第一题:奇怪的捐赠
地产大亨Q先生临终的遗愿是:拿出100万元给X社区的居民抽奖,以稍慰藉心中愧疚。
麻烦的是,他有个很奇怪的要求:
1. 100万元必须被正好分成若干份(不能剩余)。
每份必须是7的若干次方元。
比如:1元, 7元,49元,343元,...
2. 相同金额的份数不能超过5份。
3. 在满足上述要求的情况下,分成的份数越多越好!
请你帮忙计算一下,最多可以分为多少份?
思路:假如是分成10的n次方,那么就是1,10,100,1000......我们就容易知道,方法是唯一的了(那么每种的份数不能超过9),因此分成7的若干次方方法也是唯一的(每种不能超过6份),题目要求不超过5份,满足不超过6份的条件,因此方法也唯一,那么这道题的解法也就是将金额转换成7进制就可以了。
public class Main
{
public static void main(String[] args){
// 直接求一个数字的7进制表示
// 如果没有函数,可以自己用长除法取余数的方式
String s = Integer.toString(1000*1000,7);
int sum=0;
for(int i=0; i<s.length(); i++){
sum += s.charAt(i)-'0';
}
System.out.println(s);
System.out.println("result=" + sum);
}
}
第二题:天平称重
用天平称重时,我们希望用尽可能少的砝码组合称出尽可能多的重量。
如果只有5个砝码,重量分别是1,3,9,27,81
则它们可以组合称出1到121之间任意整数重量(砝码允许放在左右两个盘中)。
本题目要求编程实现:对用户给定的重量,给出砝码组合方案。
例如:
用户输入:
5
程序输出:
9-3-1
用户输入:
19
程序输出:
27-9+1
要求程序输出的组合总是大数在前小数在后。
可以假设用户的输入的数字符合范围1~121。
思路:这个题首先可以暴力,五层循环,每次都判断是放左放右还是不放
1: 1
2: 3-1
3: 3
4: 3+1
5: 9-3-1
递归解法:通过观察,5>9/2,可以总结出啥时候用减法,5=9-4,4=3+1,找到入口
数学解法:假如..只放一个盘,1,2,4,8,... 二进制,那么转换为只有0和1 的数字,0 代表不放,1代表放,这样就可以想到三进制,三进制只有0,1,2,那么问题又来了,0,1,2代表什么呢?如果是-1,0,1那就简单多了,那么我们就将2变成-1,看下图:
通过这样的方式,我们成功把一个数变成了特殊的三进制表示。
递归解法:
public class Main
{
public static void main(String[] args){
for(int i=1;i<=121;++i){
System.out.println(i+":"+f(i));
}
}
//只要超过了一半就要用减法 eg:2超出了3的一半,5超出了9的一半,14超出了27的一半
static String f(int x){
int a = 1; //最小单位
while(a<x) a *= 3;
if(a == x) return ""+a; //eg:3
if(x<=a/2) return a/3 + "+" + f(x-a/3); //eg:4
return a + reve(f(a-x)); //eg:5
}
//正负变号
static String reve(String s){
s = s.replace("-", "#");
s = s.replace("+", "-");
s = s.replace("#", "+");
return "-"+s;
}
}
进制解法:
public class Main
{
public static void main(String[] args){
for(int i=1;i<=121;++i){
System.out.println(i+":"+f(i));
}
}
static String f(int x){
String s = "";
int q = 1; //权重,代表砝码为1,3,9,27......
while(x>0){
int sh = x/3; //商
if(x%3 == 1) s = "+" + q + s;
if(x%3 == 2){
sh++;
s = "-" + q + s;
}
x = sh;
q *= 3;
}
return s.substring(1); //舍弃第一个字符(即正号)
}
}
第三题:尼姆堆
有3堆硬币,分别是3,4,5
二人轮流取硬币。
每人每次只能从某一堆上取任意数量。
不能弃权。
取到最后一枚硬币的为赢家。
求先取硬币一方有无必胜的招法。
思路:模2加法
1+1=0, 1+0=1, 0+1=1, 0+0=0
可以暴力求解法
第四题:公约公倍
如果两个数很大,怎样求最大公约数,最小公倍数?
如果是n个数呢?比如1000个数的最小公倍数
关键:算数基本定理:质因数分解的唯一性,即:每个比1大的正整数要么本身是一个素数要么可以写成一系列素数的乘积如果不考虑这些素数的在乘积中的顺序那么写出来的形式是唯一的 ,例如:14=2*7,21=3*7。
PS:欧几里得定理:gcd(a,b) = gcd(b,a%b),对a和b求最大公约数就相当于求b与(a%b)求最大公约数
原理:如果一个数n可以除开A,B,那么也一定可以除开A-B,A+B,(假设一堆苹果一堆梨,苹果可以按每堆x个正好分开,梨也可以按每堆x个正好分开,那么当其二者混在一起自然也可以按每堆x个分开,因此当一个数m可以除开A,A+B时,我们可以得到其也可以除开B,在这里我们把A+B看作A,把之前的A看作B,可以得知n也可以除开A-B,同理,n也就可以除开A-B-B,A-B-B-B...就相当于A%B)
重要公式:
最小公倍数 = 乘积 / 最大公约数
(40=5*2^3 28=7*2^2 ... 公共的部分是 2^2)
public class Main
{
public static void main(String[] args){
System.out.println("最大公约数: "+gcd(60,42));
System.out.println("最小公倍数: "+lcm(60,42));
}
//最大公约数
static int gcd(int x,int y){ //前者一定为较大的数
if(y == 0) return x; //!!!
return gcd(y,x%y);
}
//最小公倍数
static int lcm(int x,int y){
return (x*y)/gcd(x,y);
}
}
第五题:一步之遥
从昏迷中醒来,小明发现自己被关在X星球的废矿车里。
矿车停在平直的废弃的轨道上。
他的面前是两个按钮,分别写着“F”和“B”。
小明突然记起来,这两个按钮可以控制矿车在轨道上前进和后退。
按F,会前进97米。按B会后退127米。
透过昏暗的灯光,小明看到自己前方1米远正好有个监控探头。
他必须设法使得矿车正好停在摄像头的下方,才有机会争取同伴的援助。
或许,通过多次操作F和B可以办到。
矿车上的动力已经不太足,黄色的警示灯在默默闪烁...
每次进行 F 或 B 操作都会消耗一定的能量。
小明飞快地计算,至少要多少次操作,才能把矿车准确地停在前方1米远的地方。
请填写为了达成目标,最少需要操作的次数。
思路:扩展欧几里得定理:Ax + By = gcd(A,B) 有解
注意第三行:A/B是个整数,因此后面要带上A%B
第六题:有理数运算
如果求 1/2 + 1/3 + 1/4 + 1/5 + 1/6 + .... + 1/100 = ?要求绝对精确,不能有误差。
思路:方法一:BigDecimal(高精度浮点数)
方法二:一个有理数加减乘除有理数得到的还是有理数
图写错了,前面应该是加法
第七题: 素数表
第1个素数是2,第2个素数是3,...
求第100002(十万零二)个素数
思路:关键问题,数组要开多大