四种求最大公约数算法
运行最大公约数的常用算法,并进行程序的调式与测试,要求程序设计风格良好,并添加异常处理模块(如输入非法等)。
分析最大公约数的4种算法,补充完整算法, 进行程序的调式与测试,比较4种GCD算法在给定不同规模测试数据的情况下的平均运行时间,比较4种算法在不同条件下的优劣。
1.辗转相除法
辗转相除法(又名欧几里德法)C语言中用于计算两个正整数a,b的最大公约数和最小公倍数。
根据这一定理可以采用函数嵌套调用和递归调用形式进行求两个数的最大公约数和最小公倍数,现分别叙述如下:
其算法过程为: 前提:设两数为a,b设其中a 做被除数,b做除数,temp为余数
1、大数放a中、小数放b中;
2、求a/b的余数;
3、若temp=0则b为最大公约数;
4、如果temp!=0则把b的值给a、temp的值给a;
5、返回第二步;
int divisor (int a,int b) /*自定义函数求两数的最大公约数*/
{
int temp; /*定义整型变量*/
if(a
2.穷举法(利用数学定义)
穷举法(也叫枚举法)穷举法求两个正整数的最大公约数的解题步骤:从两个数中较小数开始由大到小列举,直到找到公约数立即中断列举,得到的公约数便是最大公约数 。
①定义1:对两个正整数a,b如果能在区间[a,0]或[b,0]内能找到一个整数temp能同时被a和b所整除,则temp即为最大公约数。
int divisor (int a,int b) /*自定义函数求两数的最大公约数*/
{
int temp; /*定义义整型变量*/
temp=(a>b)?b:a; /*采种条件运算表达式求出两个数中的最小值*/
while(temp>0)
{
if (a%temp==0&&b%temp==0) /*只要找到一个数能同时被a,b所整除,则中止循环*/
break;
temp--; /*如不满足if条件则变量自减,直到能被a,b所整除*/
}
return (temp); /*返回满足条件的数到主调函数处*/
}
#include "stdio.h"
main()
{
int m,n,t1;
printf("please input two integer number:");
scanf("%d%d",&m,&n);
t1=divisor(m,n);
printf("The higest common divisor is %d\n",t1);
getch();
}
3.更相减损法
更相减损术,是出自《九章算术》的一种求最大公约数的算法,它原本是为约分而设计的,但它适用于任何需要求最大公约数的场合。《九章算术》是中国古代的数学专著,其中的“更相减损术”可以用来求两个数的最大公约数,即“可半者半之,不可半者,副置分母、子之数,以少减多,更相减损,求其等也。以等数约之。”更相减损术,是出自《九章算术》的一种求最大公约数的算法,它原本是为约分而设计的,但它适用于任何需要求最大公约数的场合。《九章算术》是中国古代的数学专著,其中的“更相减损术”可以用来求两个数的最大公约数,即“可半者半之,不可半者,副置分母、子之数,以少减多,更相减损,求其等也。以等数约之。” 翻译成现代语言如下: 第一步:任意给定两个正整数;判断它们是否都是偶数。若是,则用2约简;若不是则执行第二步。 第二步:以较大的数减较小的数,接着把所得的差与较小的数比较,并以大数减小数。继续这个操作,直到所得的减数和差相等为止。 则第一步中约掉的若干个2与第二步中等数的乘积就是所求的最大公约数。 其中所说的“等数”,就是最大公约数。求“等数”的办法是“更相减损”法。所以更相减损法也叫等值算法。
int gcd(int m,int n)
{
int i=0,temp,x;
while(m%2==0 && n%2==0) //判断m和n能被多少个2整除
{
m/=2;
n/=2;
i+=1;
}
if(mx)?n:x;
n=(n
4.Stein算法
Stein算法由J. Stein 1961年提出,这个方法也是计算两个数的最大公约数。来研究一下最大公约数的性质,发现有 gcd( kx,ky ) = k*gcd( x,y ) 这么一个非常好的性质。试取 k=2,则有 gcd( 2x,2y ) = 2 * gcd( x,y )。很快联想到将两个偶数化小的方法。那么一奇一个偶以及两个奇数的情况如何化小呢?
先来看看一奇一偶的情况: 设有2x和y两个数,其中y为奇数。因为y的所有约数都是奇数,所以 a = gcd( 2x,y ) 是奇数。根据2x是个偶数不难联想到,a应该是x的约数。我们来证明一下:(2x)%a=0,设2x=na,因为a是奇数,2x是偶数,则必有n是偶数。又因为 x=(n/2)a,所以 x%a=0,即a是x的约数。因为a也是y的约数,所以a是x和y的公约数,有 gcd( 2x,y ) <= gcd( x,y )。因为gcd( x,y )明显是2x和y的公约数,又有gcd( x,y ) <= gcd( 2x,y ),所以 gcd( 2x,y ) = gcd( x,y )。至此,我们得出了一奇一偶时化小的方法。
再来看看两个奇数的情况:设有两个奇数x和y,不妨设x>y,注意到x+y和x-y是两个偶数,则有 gcd( x+y,x-y ) = 2 * gcd( (x+y)/2,(x-y)/2 ),那么 gcd( x,y ) 与 gcd( x+y,x-y ) 以及 gcd( (x+y)/2,(x-y)/2 ) 之间是不是有某种联系呢?为了方便设 m=(x+y)/2 ,n=(x-y)/2 ,容易发现 m+n=x ,m-n=y 。设 a = gcd( m,n ),则 m%a=0,n%a=0 ,所以 (m+n)%a=0,(m-n)%a=0 ,即 x%a=0 ,y%a=0 ,所以a是x和y的公约数,有 gcd( m,n )<= gcd(x,y)。再设 b = gcd( x,y )肯定为奇数,则 x%b=0,y%b=0 ,所以 (x+y)%b=0 ,(x-y)%b=0 ,又因为x+y和x-y都是偶数,跟前面一奇一偶时证明a是x的约数的方法相同,有 ((x+y)/2)%b=0,((x-y)/2)%b=0 ,即 m%b=0 ,n%b=0 ,所以b是m和n的公约数,有 gcd( x,y ) <= gcd( m,n )。所以 gcd( x,y ) = gcd( m,n ) = gcd( (x+y)/2,(x-y)/2 )。
整理一下,对两个正整数 x>y :
1.均为偶数 gcd( x,y ) =2gcd( x/2,y/2 );
2.均为奇数 gcd( x,y ) = gcd( (x+y)/2,(x-y)/2 );
2.x奇y偶 gcd( x,y ) = gcd( x,y/2 );
3.x偶y奇 gcd( x,y ) = gcd( x/2,y ) 或 gcd( x,y )=gcd( y,x/2 );
现在已经有了递归式,还需要再找出一个退化情况。注意到 gcd( x,x ) = x ,就用这个。
int Stein( unsigned int x, unsigned int y )
/* return the greatest common divisor of x and y */
{
int factor = 0;
int temp;
if ( x < y )
{
temp = x;
x = y;
y = temp;
}
if ( 0 == y )
{
return 0;
}
while ( x != y )
{
if ( x & 0x1 )
{/* when x is odd */
if ( y & 0x1 )
{/* when x and y are both odd */
y = ( x - y ) >> 1;
x -= y;
}
else
{/* when x is odd and y is even */
y >>= 1;
}
}
else
{/* when x is even */
if ( y & 0x1 )
{/* when x is even and y is odd */
x >>= 1;
if ( x < y )
{
temp = x;
x = y;
y = temp;
}
}
else
{/* when x and y are both even */
x >>= 1;
y >>= 1;
++factor;
}
}
}
return ( x << factor );
}
自己写的公共主函数:
#include "stdio.h" /*输入输出类头文件*/
#include
#include
int name[1024];//存取数据的数组
//手动输入函数
int selfscan()
{
printf("请输入数据:\n"); //输入
for(int i=0;i<100;i++)
{
if(i>0)
{
while(name[i-1]<0) //输入小于0的数报错,重新输入
{printf("error:");scanf("%d",&name[i-1]);}
if(name[i-1]==0) //输入0,结束输入
{return 0;}
}
scanf("%d",&name[i]);
}
return 0;
}
//随机数产生
int scan()
{
srand(1);
int i=0;
for(i=0;i<100;)
name[i++]=rand()%34;
return 0;
}
//打印数组
int print()
{
int l=0;
printf("输入的数据为:\n");//打印数组
for(;l<100;)
{
if(name[l]==0)
{printf("\n");
return 0;}
printf("%4d",name[l++]);
if(l%10==0)
{printf("\n");}
}
printf("\n");
return 0;
}
int print2() //随机数出现0时,继续打印
{
int l=0;
printf("输入的数据为:\n");//打印数组
for(;l<100;)
{
printf("%4d",name[l++]);
if(l%10==0)
{printf("\n");}
}
printf("\n");
return 0;
}
void main()
{
clock_t start,stop;
double Total_time;
int di,mu;//di为公约数返回值,mu为公倍数返回值
int j=0,k=1;//j,k用来计数,数组输出用
int flag=0;//flag标记是否自己输入(0为自己输入)
printf("请选择 自己输入 还是 随机输入(自己输入请按0,随机按1):");
fflush(stdin);
scanf("%d",&flag);
if(flag==0)
{selfscan(); //自己输入数据函数
print();} //打印输入数据函数
else if(flag==1)
{scan(); //随机输入数据函数
print2();} //打印输入数据函数
start=clock();//开始计时
for(;j<100;j++,k++) //计算并打印
{
if((name[j]==0)&&(name[k]==0))
break;
fflush(stdin);
di=divisor(name[j],name[k]);
mu=multiple(name[j],name[k]);
printf("%8d和%6d的最大公约数为: %8d ",name[j],name[k],di);
printf("\t最小公倍数为: %8d \n\n",mu);
}
stop=clock();//结束计时
Total_time=((double)(stop-start))/CLK_TCK;
printf("运算时间:%lf\n",Total_time); //打印时间
}
简单百组数据测试:
1.辗转相除法
时间复杂度:对数级
2.穷举法(利用数学定义)
3.更相减损法
4.Stein算法
百组数据分析,辗转相除法相较其他算法耗费时间更多,更慢。Stein算法有事不明显,但Stein算法是通过移位计算,对于数据数值特别大时,能更快的算出结果,对于这些几位数运算不明显。
刚开始数的输入使用scanf函数直接输入,只能进行一次计算,后边使用数组输入,多次调取。但数组输入每次都要输入满才能进行下一个函数,要不未输入的地方仍会进行运算,特意设置输入为0时结束,打印输出为0结束打印,当遇到输入为0时结束最大公倍数计算,结束打印结果。为使避免手工输入,避免麻烦,使用伪随机函数,产生随机数,而且为伪随机函数输入相同时,产生的数相同,可以用来比较这几个最大公倍数函数的运行结果速度。人工输入可能输入错误,输入负数,(0为结束输入)所以加入判定输入函数,为负数时,输出error,提示重新输入,下次输入覆盖这次输入。但伪随机数产生可能会出现0,然后结束打印数组,所以修改原打印函数创建新函数,声明flag标记判定是人工输入还是随机产生,然后使用对应的打印函数。