一、题目
形如:1/a 的分数称为单位分数。
可以把1分解为若干个互不相同的单位分数之和。
例如:
1 = 1/2 + 1/3 + 1/9 + 1/18
1 = 1/2 + 1/3 + 1/10 + 1/15
1 = 1/3 + 1/5 + 1/7 + 1/9 + 1/11 + 1/15 + 1/35 + 1/45 + 1/231
等等,类似这样的分解无穷无尽。
我们增加一个约束条件:最大的分母必须不超过30
请你求出分解为n项时的所有不同分解法。
数据格式要求:
输入一个整数n,表示要分解为n项(n<12)
输出分解后的单位分数项,中间用一个空格分开。
每种分解法占用一行,行间的顺序按照分母从小到大排序。
例如,
输入:
4
程序应该输出:
1/2 1/3 1/8 1/24
1/2 1/3 1/9 1/18
1/2 1/3 1/10 1/15
1/2 1/4 1/5 1/20
1/2 1/4 1/6 1/12
二、分析
假如1被分为了四项,即1 —->1/2 + 1/3 + 1/9 + 1/18,如何判断1和它是否相等?
因为分数的求解不容易,可以这样转为整数的形式,即判断2*3*9*18是否等于3*9*18+2*9*18+2*3*18+2*3*9,
如果相等,则说明1==1/2+1/3+1/9+1/18。
题目中要求
1、最大的分母必须不超过30
2、每种分解法占用一行,行间的顺序按照分母从小到大排序
创建一个数字a,它用于存储每一项的分母,这样就容易判断分母不超过30,且按照从小到大顺序排序输出。
三、代码
public class Demo6 {
/**
*
* @param num
* Max项数
* @param a
* 存放分母的数组
* @param k
* 从0到num
*/
public static void getResult(int num, int a[], int k) {
if (num == k) {// 数组a赋值结束,判断是否符合条件
judge(a);
return; // 跳出此次函数到上一个f函数,重新对a进行赋值
}
for (int i = 2; i < 30; i++) {// 分母最大为30
a[k] = i;
getResult(num, a, k + 1);
}
}
//判断是否满足题目的条件
public static void judge(int a[]) {
int rightSum = 1;
int smallSum = 1;
int leftSum = 0;
// 筛选等于,后面大于前面的数---要求按分母从小到大
for (int k = 0; k < a.length; k++) {
for (int l = k + 1; l < a.length; l++) {
if (a[k] >= a[l]) {
return;
}
}
}
// 判断分母数组是否符合要求
// 1/a+1/b+1/c==1
// a*c+b*c+a*b==a*b*c
for (int i = 0; i < a.length; i++) {
rightSum = rightSum * a[i];
for (int j = 0; j < a.length; j++) {
if (i == j) {
continue;
}
smallSum = smallSum * a[j];
}
leftSum = leftSum + smallSum;
smallSum = 1; // smallSum需要清1
}
// 说明满足找到满足条件的分解方法,输出
if (rightSum == leftSum) {
print(a);
} else {
return;
}
}
// 打印
public static void print(int a[]) {
System.out.println();
for (int i = 0; i < a.length; i++) {
if (i == a.length - 1) {
System.out.print("1/" + a[i] + " ");
} else {
System.out.print("1/" + a[i] + "+");
}
}
}
public static void main(String[] args) {
Scanner input = new Scanner(System.in);
int num = input.nextInt();
int a[] = new int[num];
int k = 0;
getResult(num, a, k);
}
}