花朵数

【编程大题】

花朵数一个 N 位的十进制正整数,如果它的每个位上的数字的 N 次方的和等于这个数本身,则称其为花朵数。例如:当 N=3 时,153 就满足条件,因为 1^3 + 5^3 + 3^3 = 153,这样的数字也被称为水仙花数

(其中,“^”表示乘方,5^3 表示 5 的 3 次方,也就是立方)。

当 N=4 时,1634 满足条件,因为 1^4 + 6^4 + 3^4 + 4^4 = 1634。

当 N=5 时,92727 满足条件。实际上,对 N 的每个取值,可能有多个数字满足条件。

程序的任务是:求 N=21 时,所有满足条件的花朵数。

注意:这个整数有 21 位,它的各个位数字的 21 次方之和正好等于这个数本身。

如果满足条件的数字不只有一个,请从小到大输出所有符合条件的数字,每个数字占一行。

因为这个数字很大,请注意解法时间上的可行性。

要求程序在 1 分钟内运行完毕。【程序运行参考结果】

128468643043731391252

449177399146038697307

【解题思路】

方法一:一般看到题目想到的是从21位数中最小数循环到最大数,依次计算每个数的21次方,再相加判断。

瓶颈:

1.超过基本数据类型int和long型的范围,可以使用大数操作BigInteger(详情参考:https://blog.youkuaiyun.com/zhongkelee/article/details/52289163)

2.计算量太大,计算机无法处理。

方法二:因为需要用到0~9的21次方,所以提前计算出来,存储到数组,减少计算量。虽然是21位数,但是根据题目,各个数字之间是可以没有顺序的,所以把21个名额中0~9出现的次数计算出来,并求得21个名额的各个数的21次方和,最后判断和中出现0~9的次数与21个名额中0~9的次数是否一致,完全符合则满足题目要求。

package com.sise.test;

import java.math.BigInteger;

public class Test01 {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		
		//0~9每个数字的21次方
		BigInteger[] pw={p(0),p(1),p(2),p(3),p(4),p(5),p(6),p(7),p(8),p(9)};
		
		//0~9每个数字在21位数中出现的次数
		int[] nn=new int[10];
		
		//算法核心:21个名额中:0~9每个数字出现的次数进行“大排列”
		//第三个数:当前试验数字的次数;第四个数:当前名额占用了多少
		f(pw,nn,0,0);
		
	}

	//求x的21次方
	public static BigInteger p(int x) {
		// TODO Auto-generated method stub
		BigInteger base=BigInteger.ONE;
		for(int i=0;i<21;i++){
			base=base.multiply(BigInteger.valueOf(x));
		}
		return base;
	}

	public static void f(BigInteger[] pw, int[] nn, int cur, int use) {
		// TODO Auto-generated method stub
		//选到了最后一个数字
		if(cur==9){
			nn[9]=21-use;
			Calculate(pw,nn);
			return;
		}
		
		//对当前位置所有可能进行枚举
		for(int i=0;i<=21-use;i++){
			nn[cur]=i;
			f(pw,nn,cur+1,use+i);
		}
	}

	public static void Calculate(BigInteger[] pw, int[] nn) {
		// TODO Auto-generated method stub
		
		BigInteger sum=BigInteger.ZERO;
		for(int i=0;i<10;i++){
			sum=sum.add(pw[i].multiply(BigInteger.valueOf(nn[i])));
		}
		
		String s=""+sum;
		if(s.length()!=21)return;
		
		//确定和中各个数字出现多少次
		int[] nn2=new int[10];
		
		for(int i=0;i<21;i++){
			nn2[s.charAt(i)-'0']++;
		}
		
		for(int i=0;i<10;i++){
			if(nn[i]!=nn2[i]){
				return;
			}
		}
		
		//完全匹配!打印结果
		System.out.println(s);
	}
	
}

具体视频讲解http://v.youku.com/v_show/id_XNDQ0OTg2NjYw.html

 (编程题)花朵 一个N位的十进制正整,如果它的每个位上的字的N次方的和等于这个本身,则称其为花朵。 例如: 当N=3时,153就满足条件,因为 1^3 + 5^3 + 3^3 = 153,这样的字也被称为水仙花(其中,“^”表示乘方,5^3表示5的3次方,也就是立方)。 当N=4时,1634满足条件,因为 1^4 + 6^4 + 3^4 + 4^4 = 1634。 当N=5时,92727满足条件。 实际上,对N的每个取值,可能有多个字满足条件。 程序的任务是:求N=21时,所有满足条件的花朵。注意:这个整有21位,它的各个位字的21次方之和正好等于这个本身。 如果满足条件的字不只有一个,请从小到大输出所有符合条件的字,每个字占一行。因为这个字很大,请注意解法时间上的可行性。要求程序在3分钟内运行完毕。 【程序运行参考结果】 128468643043731391252 449177399146038697307  (编程题)信用卡号验证 当你输入信用卡号码的时候,有没有担心输错了而造成损失呢?其实可以不必这么担心,因为并不是一个随便的信用卡号码都是合法的,它必须通过Luhn算法来验证通过。 该校验的过程: 1、从卡号最后一位字开始,逆向将奇位(1、3、5等等)相加。 2、从卡号最后一位字开始,逆向将偶字,先乘以2(如果乘积为两位,则将其减去9),再求和。 3、将奇位总和加上偶位总和,结果应该可以被10整除。 例如,卡号是:5432123456788881 则奇、偶位(用红色标出)分布:5432123456788881 奇位和=35 偶位乘以2(有些要减去9)的结果:1 6 2 6 1 5 7 7,求和=35。 最后35+35=70 可以被10整除,认定校验通过。 请编写一个程序,从标准输入获得卡号,然后判断是否校验通过。通过显示:“成功”,否则显示“失败”。 比如,用户输入:356827027232780 程序输出:成功 【程序测试参考用例】 356406010024817 成功 358973017867744 成功 356827027232781 失败 306406010024817 失败 358973017867754 失败  (c/c++组代码填空)下列代码把一个串p复制到新的位置q。请填写缺少的语句; char* p = "abcde"; char* q = (char*)malloc(strlen(p)+1); for(int i=0;_____________; i++) q[i] = p[i]; q[i] = 0; 【参考答案】 P[i] 或 *(p+i) 或 p[i] != '\0' 或 ... (答案不唯一,以选手提供代码带入专用验证程序测试为依据,验证程序比题面中提供的程序片段更严谨,更完善)  (Java组代码填空)有n个孩子站成一圈,从第一个孩子开始顺时针方向报,报到3的人出列,下一个人继续从1报,直到最后剩下一个孩子为止。问剩下第几个孩子。下面的程序以10个孩子为例,模拟了这个过程,请完善之(提示:报的过程被与之逻辑等价的更容易操作的过程所代替)。 Vector a = new Vector(); for(int i=1; i<=10; i++) { a.add("第" + i + "个孩子"); } for(;;) { if(a.size()==1) break; for(int k=0; k<2; k++) ________________; a.remove(0); } System.out.println(a); 【参考答案】 a.add(a.remove(0)) (答案不唯一,以选手提供代码带入专用验证程序测试为依据,验证程序比题面中提供的程序片段更严谨,更完善)  (结果填空题) 有趣的平方 625这个字很特别,625的平方等于390625,刚好其末3位是625本身。除了625,还有其它的3位有这个特征吗?还有一个!该是:_____________ 【参考答案】 376
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值