小朋友学编程—求最大公约数

一、最大公约数(Greatest Common Divisor)

几个自然数,公有的因数,叫做这几个数的公约数;其中最大的一个,叫做这几个数的最大公约数。例如:18、12的公约数有1、2、3、6,其中最大的一个是6,6是18与12的最大公约数,一般记为(18、12)=6。

二、编程求最大公约数

用c++编程实现,输入任意两个自然数a和b,求他们的最大公约数。
这个题目主要是考察小朋友们对循环的使用。

三、算法求解

1.穷举法

解析

穷举法是大部分人最先想到的方法。让一个数循环的去被a和b除,如果余数都为0那么就是他们的公约数,然后最大的那个就是最大公约数。

代码

#include <iostream>
using namespace std;

//穷举法
int gcd1(int a, int b)
{
    int x=(a<b?a:b);
    int z = 0 ,count=0;
    for(;x>0;x--)
    {
        if( a%x == 0 && b%x == 0 ){
            z=x;
            break;
        }
        count++;
    }
    cout<<"循环次数(穷举法):"<<count<<endl;
    return z;
}

int main(){
    int a,b,result;
    cin>>a>>b;
    result = gcd1(a,b);
    cout<<a<<"和"<<b<<"的最大公约数:"<<result<<endl;
    return 0;
}

运行结果:

18 12
循环次数(穷举法):6
18和12的最大公约数:6

分析

穷举法虽然简单,但是有一个很大的缺点,就是效率低。比如咱们输入1200和1800,那么程序是从1200开始自减,一直减到600,才得出了结果。这个过程for共执行了600次。含有很多小朋友会从1开始写自增,那么循环的次数就更多了。
所以求最大公约数,通常不用穷举法。
穷举法的效率极其的低下,和后面其他几个不在一个数量级上。

2.辗转相除法

解析

辗转相除法,又称欧几里得算法。用于计算两个正整数a,b的最大公约数和最小公倍数,其依赖于gcd(a,b) = a (b=0)和gcd(b,a mod b) (b!=0)。算法的具体描述如下图:


辗转相除法图示

代码

#include <iostream>
using namespace std;

//迭代相除法
int gcd2(int a, int b)
{
    int z = b;
    int count = 0;
    while (a % b != 0) {
        z = a % b;
        a = b;
        b = z;
        count++;
    }
    cout<<"循环次数(迭代相除):"<<count<<endl;
    return z;
}

int main(){
    int a,b,result;
    cin>>a>>b;
    result = gcd2(a,b);
    cout<<a<<"和"<<b<<"的最大公约数:"<<result<<endl;
    return 0;
}

运行结果:

18 12
循环次数(迭代相除):1
18和12的最大公约数:6

分析

可以看到迭代相除只用了一次循环就得到了结果,大大提高了效率。
此方法当数据较小的时候性能最好,代码可读性极好,但是它需要用到取余运算.

扩展

小朋友们只学到了循环,如果学到函数以及递归则可以把函数写成如下这样,大大简化代码提高可读性。

int gcd(int a, int b)
{
    return 0 == b ? a : gcd(b, a % b);
}

3.更相减损法

解析

更相减损术是出自《九章算术》的一种求最大公约数的算法,它原本是为约分而设计的,但它适用于任何需要求最大公约数的场合。
九章算术》是中国古代的数学专著,其中的“更相减损术”可以用来求两个数的最大公约数,原文是:
可半者半之,不可半者,副置分母、子之数,以少减多,更相减损,求其等也。以等数约之。
白话文译文:
(如果需要对分数进行约分,那么)可以折半的话,就折半(也就是用2来约分)。如果不可以折半的话,那么就比较分母和分子的大小,用大数减去小数,互相减来减去,一直到减数与差相等为止,用这个相等的数字来约分。
算法的图示如下:

更相减损法

代码

#include <iostream>
using namespace std;

//更相减损法
int gcd3(int a,int b)
{
    int count = 0;
    while(a != b)
    {
        if(a>b)
        {
            a = a - b;
        }
        else
        {
            b = b - a;
        }
        count++;
    }
    cout<<"循环次数(更相减损法):"<<count<<endl;
    return a;
}

int main(){
    int a,b,result;
    cin>>a>>b;
    result = gcd3(a,b);
    cout<<a<<"和"<<b<<"的最大公约数:"<<result<<endl;
    return 0;
}

运行结果:

18 12
循环次数(更相减损法):2
18和12的最大公约数:6

分析

可以看到更相减损法用了两次循环得到结果,效率也挺高,还有它不需要取余。
这种方法在计算大数的情况下依旧可以保持较快的速度,代码的可读性也非常高,但是因为要不断互减,在两个数较为接近的时候需要的系统资源较大。

4.短除法

解析

短除法应该是小朋友最熟悉的数学方法,和计算数学题时常采用的方法一致。


短除法

左边部分的因子相乘,即为最大公约数。
所以,18与12的最大公约数为2 * 3 = 6

// 短除法
int gcd4(int a, int b)
{
    int min = a < b ? a : b;
    int s = 1;
    int i;
    for(i = 2; i <= min ; i++)
    {
        // 四个条件只要有一个不满足,while循环结束
        while(a > 0 && b > 0 && 0 == a % i && 0 == b % i)
        {
            a /= i;
            b /= i;
            s *= i;
        }
    }
    return s;
}

int main(){
    int a,b,result;
    cin>>a>>b;
    result = gcd4(a,b);
    cout<<a<<"和"<<b<<"的最大公约数:"<<result<<endl;
    return 0;
}

分析

短除法的效率也还算可以,它更多是和我们常用的数学方法一致,算法上比较容易理解,但是代码的可读性就比较差一些。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,458评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,454评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,171评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,062评论 0 207
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,440评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,661评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,906评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,609评论 0 200
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,379评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,600评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,085评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,409评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,072评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,088评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,860评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,704评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,608评论 2 270

推荐阅读更多精彩内容