算法设计分析: 埃及分数问题
2014-05-26 16:12
696 查看
埃及分数问题
时间限制:2000MS 内存限制:1000K
描述
所为埃及分数,是指分子为1的分数。任何一个分数都可以表示为埃及分数之和的形式。如:2/3=1/2+1/6,但不允许2/3=1/3+1/3,因为加数中有相同的。
对于一个分数a/b,表示方法有很多种,但是哪种最好呢?
首先,加数少的比加数多的好,其次,加数个数相同的,最小的分数越大越好。
如:
19/45=1/3 + 1/12 + 1/180
19/45=1/3 + 1/15 + 1/45
19/45=1/3 + 1/18 + 1/30
19/45=1/4 + 1/6 + 1/180
19/45=1/5 + 1/6 + 1/18.
最好的是最后一种,因为1/18比1/180,1/45,1/30,1/180都大。
对于给定的真分数,设计一个算法,找到用最好埃及分数表示真分数的表达式。
输入格式
仅一行,包括两个整数a和b,它们之间用空格分开,分别表示分数的分子和分母。
输出格式
用最好的埃及分数表示法来表示分数中的分母,从小到大依次输出。
输入样例
19 45
输出样例
5 6 18
Hint
按a/b可拆分成的单位分数的个数进行循环搜索。
1,如果a=1则直接输出。
2,否则从2个开始尝试。如果2个单位分数无法拼凑成a/b,则试3个……依此类推。那么我们搜索得到的第一个解一定会满足加数个数最少这个要求。
3,如果当加数个数等于n的时候搜索到一组解,那么只要在个数为n的这次搜索中选择最优解(分母最小)。
#include"stdio.h"
int temp[10000] = {0};
int best[10000] = {0};
bool answer = false;
void divide(int x,int y,int m,int n){
int k;
if(m == n){
if(y % x == 0 && x * temp[n - 1] < y){//(1 / temp[n-1]) > (y / x)
temp
= y / x;
answer = true;//找到最小个数,令 n 不再 加深
k = n;
if(best[k] == 0){//n-1个数不能满足要求 ,best[1···n]被全部 覆盖
for(int i = 1; i <= n; i++)best[i] = temp[i];
return;
}
while(true){
if(temp[k] > best[k])break;
if(temp[k] == best[k]){
k--;
continue;
}
for(int i = 1; i <= n; i++)best[i] = temp[i];
break;
}
}
}
else {
int max = (n - m + 1) * y / x;//不能大于剩下的 (n - m + 1 )等份
int min = (y / x ) > (temp[m - 1] + 1)? y / x : temp[m - 1] + 1;
for(int i = min; i <= max; i++){
temp[m] = i;
if(x * i - y < 0)
continue;
else {
if(best[m] == 0)best[m] = temp[m];
divide(x * i - y, y * i,m + 1,n);
}
}
}
}
int main(){
int x,y,count = 1;
scanf("%d%d",&x,&y);
do{
divide(x,y,1,count++);
}while(!answer);
for(int i = 1; i < count; i++)
printf("%d ",best[i]);
return 1;
}
时间限制:2000MS 内存限制:1000K
描述
所为埃及分数,是指分子为1的分数。任何一个分数都可以表示为埃及分数之和的形式。如:2/3=1/2+1/6,但不允许2/3=1/3+1/3,因为加数中有相同的。
对于一个分数a/b,表示方法有很多种,但是哪种最好呢?
首先,加数少的比加数多的好,其次,加数个数相同的,最小的分数越大越好。
如:
19/45=1/3 + 1/12 + 1/180
19/45=1/3 + 1/15 + 1/45
19/45=1/3 + 1/18 + 1/30
19/45=1/4 + 1/6 + 1/180
19/45=1/5 + 1/6 + 1/18.
最好的是最后一种,因为1/18比1/180,1/45,1/30,1/180都大。
对于给定的真分数,设计一个算法,找到用最好埃及分数表示真分数的表达式。
输入格式
仅一行,包括两个整数a和b,它们之间用空格分开,分别表示分数的分子和分母。
输出格式
用最好的埃及分数表示法来表示分数中的分母,从小到大依次输出。
输入样例
19 45
输出样例
5 6 18
Hint
按a/b可拆分成的单位分数的个数进行循环搜索。
1,如果a=1则直接输出。
2,否则从2个开始尝试。如果2个单位分数无法拼凑成a/b,则试3个……依此类推。那么我们搜索得到的第一个解一定会满足加数个数最少这个要求。
3,如果当加数个数等于n的时候搜索到一组解,那么只要在个数为n的这次搜索中选择最优解(分母最小)。
#include"stdio.h"
int temp[10000] = {0};
int best[10000] = {0};
bool answer = false;
void divide(int x,int y,int m,int n){
int k;
if(m == n){
if(y % x == 0 && x * temp[n - 1] < y){//(1 / temp[n-1]) > (y / x)
temp
= y / x;
answer = true;//找到最小个数,令 n 不再 加深
k = n;
if(best[k] == 0){//n-1个数不能满足要求 ,best[1···n]被全部 覆盖
for(int i = 1; i <= n; i++)best[i] = temp[i];
return;
}
while(true){
if(temp[k] > best[k])break;
if(temp[k] == best[k]){
k--;
continue;
}
for(int i = 1; i <= n; i++)best[i] = temp[i];
break;
}
}
}
else {
int max = (n - m + 1) * y / x;//不能大于剩下的 (n - m + 1 )等份
int min = (y / x ) > (temp[m - 1] + 1)? y / x : temp[m - 1] + 1;
for(int i = min; i <= max; i++){
temp[m] = i;
if(x * i - y < 0)
continue;
else {
if(best[m] == 0)best[m] = temp[m];
divide(x * i - y, y * i,m + 1,n);
}
}
}
}
int main(){
int x,y,count = 1;
scanf("%d%d",&x,&y);
do{
divide(x,y,1,count++);
}while(!answer);
for(int i = 1; i < count; i++)
printf("%d ",best[i]);
return 1;
}
相关文章推荐
- 算法设计与分析--N皇后问题实现程…
- 用递归法:设计算法求解汉诺塔问题,并编程实现。 (1) Hanoi(汉诺)塔问题分析 这是一个古典的数学问题,是一个用递归方法解题的典型例子。问题是这样的:古代有一个梵塔,塔内有3个座 A,B,C
- 设计算法,在O(n)时间内求解分数背包问题
- 算法设计与分析题目练习三:骑士旅游问题(回溯算法)
- 计算机算法设计与分析作业01:分治法求解大数乘法+L型骨牌的棋盘覆盖问题
- 【算法分析与设计】全排列问题
- (算法竞赛入门)埃及分数问题学习笔记
- 计算机算法设计与分析之棋盘覆盖问题
- 【算法设计与分析】递归与分治----2.4 排列问题
- 算法分析与设计-15-背包问题的贪心算法
- 算法设计与分析--小算法问题总结
- 动态规划法之投资问题_算法分析与设计
- 【算法设计与分析】2、棋盘覆盖问题
- 设计一个算法,把一个真分数表示为埃及分数之和的形式。所谓埃及分数是指分子为1的分数。如7、8=1/2+1/3+1/24。
- 算法分析与设计矩阵连乘问题
- 计算机算法设计与分析之棋盘覆盖问题
- 【算法设计与分析】最大子段和问题
- 算法竞赛入门:埃及分数问题
- 算法分析与设计-09- 0-1背包问题
- 【算法设计与分析】输油管道问题