求最大公約數函數gcd

在計算機編程中,求最大公約數是一件很常見的事情。最大公約數是指兩個或多個整數共有約數中,最大的一個數。而求最大公約數的函數gcd也成為編程中的常用演算法之一。

一、gcd函數的定義與基本思路

int gcd(int a, int b) {
    while (b != 0) {
        int r = a % b;
        a = b;
        b = r;
    }
    return a;
}

上述代碼實現了求a和b的最大公約數的函數gcd。其基本思路是不斷地將a,b作為除數,r= a % b的餘數作為除數,直到b變成0,此時a就是最大公約數。

當然,上述代碼也可以寫為遞歸形式:

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

兩種寫法其實是原理相同,只是實現方式不同而已。

二、優化演算法

前面的演算法定義簡潔,思路明晰,但是卻存在一個慢的問題。

我們看一下,比如求100和87的最大公約數,這裡大家可以手算一下,結果是1。而如果用前面的演算法來求,需要循環100次(即b從1到100),才能得到最後的結果。當然,這只是比較複雜情況下的一種極端情況。但是可想而知,若要計算更大數值的最大公約數,則這種循環演算法需要的時間將更長。

這時我們可以用更高效率的輾轉相除演算法——歐幾里得演算法。

歐幾里得演算法是基於這樣一條性質:兩個整數的最大公約數等於其中較小的數和兩數相除餘數的最大公約數。

比如如下的例子,用輾轉相除法求最大公約數:

gcd(319,377)=gcd(319,58)=gcd(49,58)=gcd(49,9)=gcd(4,9)=gcd(4,1)=4

這裡的思路是不斷把較大的那個數減去較小的數,直到兩者相等,那麼這個數就是最大公約數。

有了這個演算法,代碼實現起來就非常簡單了。

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

三、gcd函數的應用

除了求最大公約數以外,gcd函數還有很多應用,下面我們列舉幾個典型的。

1. 最小公倍數

最小公倍數定義為兩個或多個整數公有的倍數中,最小的一個整數。最小公倍數等於兩個整數的乘積除以它們的最大公約數。

int lcm(int a, int b) {
    return a * b / gcd(a, b);
}

2. 分數的化簡

在一些涉及到分數情況下,我們需要把分式化簡為最簡形式。對於一個分數a/b,其最簡形式即分子分母沒有其他公因數存在。那麼,我們只需要調用求a和b的最大公約數函數,然後分子和分母同除以最大公約數即可。

int fenzi,fenmu;//定義分子和分母
int gcd_value = gcd(fenzi,fenmu);//求最大公約數
fenzi /=gcd_value;//分子除以最大公約數
fenmu /=gcd_value;//分母除以最大公約數

3. 求解線性同餘方程

線性同餘方程是指形如ax ≡ b(mod m)的方程,其中a、x、b、m均為整數。

一個典型的應用是求解模線性方程ax + by = c 的解(x,y)。

其中,c是常數,a、b是整數(a,b同樣也是線性同餘方程的係數),x、y是整數解。

解法是首先求出a、b的最大公約數,如果c不是其的倍數則無整數解,否則用擴展歐幾里得演算法求得x、y的一組通解,在此基礎上計算出特解。

bool solve(int a, int b, int c, int &x,int &y){
    int d=gcd(a,b);
    if(c%d) return false;
    x=y=0;
    if(d==a){
        x=c/d;
        return true;
    } 
    if(d==b){
        y=c/d;
        return true;
    }
    int k=c/d;
    if(solve(b, a%d, c%d, x, y)){
        x+=k;
        swap(x,y);
        return true;
    }
    return false;
}

小結

最大公約數是一類非常基礎的演算法問題,在我們實際編程中也經常會用到,如求最小公倍數,化簡分數等。這裡我們介紹了兩種求最大公約數的方法,分別是輾轉相除法和歐幾里得演算法,在實際應用中可以根據需要選擇不同的方法進行求解。同時,我們還看到了求解線性同餘方程的應用。

原創文章,作者:IDFEK,如若轉載,請註明出處:https://www.506064.com/zh-tw/n/333803.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
IDFEK的頭像IDFEK
上一篇 2025-02-01 13:34
下一篇 2025-02-01 13:34

相關推薦

  • Python中引入上一級目錄中函數

    Python中經常需要調用其他文件夾中的模塊或函數,其中一個常見的操作是引入上一級目錄中的函數。在此,我們將從多個角度詳細解釋如何在Python中引入上一級目錄的函數。 一、加入環…

    編程 2025-04-29
  • Python中capitalize函數的使用

    在Python的字元串操作中,capitalize函數常常被用到,這個函數可以使字元串中的第一個單詞首字母大寫,其餘字母小寫。在本文中,我們將從以下幾個方面對capitalize函…

    編程 2025-04-29
  • Python中set函數的作用

    Python中set函數是一個有用的數據類型,可以被用於許多編程場景中。在這篇文章中,我們將學習Python中set函數的多個方面,從而深入了解這個函數在Python中的用途。 一…

    編程 2025-04-29
  • 單片機列印函數

    單片機列印是指通過串口或並口將一些數據列印到終端設備上。在單片機應用中,列印非常重要。正確的列印數據可以讓我們知道單片機運行的狀態,方便我們進行調試;錯誤的列印數據可以幫助我們快速…

    編程 2025-04-29
  • 三角函數用英語怎麼說

    三角函數,即三角比函數,是指在一個銳角三角形中某一角的對邊、鄰邊之比。在數學中,三角函數包括正弦、餘弦、正切等,它們在數學、物理、工程和計算機等領域都得到了廣泛的應用。 一、正弦函…

    編程 2025-04-29
  • Python3定義函數參數類型

    Python是一門動態類型語言,不需要在定義變數時顯示的指定變數類型,但是Python3中提供了函數參數類型的聲明功能,在函數定義時明確定義參數類型。在函數的形參後面加上冒號(:)…

    編程 2025-04-29
  • Python定義函數判斷奇偶數

    本文將從多個方面詳細闡述Python定義函數判斷奇偶數的方法,並提供完整的代碼示例。 一、初步了解Python函數 在介紹Python如何定義函數判斷奇偶數之前,我們先來了解一下P…

    編程 2025-04-29
  • Python實現計算階乘的函數

    本文將介紹如何使用Python定義函數fact(n),計算n的階乘。 一、什麼是階乘 階乘指從1乘到指定數之間所有整數的乘積。如:5! = 5 * 4 * 3 * 2 * 1 = …

    編程 2025-04-29
  • 分段函數Python

    本文將從以下幾個方面詳細闡述Python中的分段函數,包括函數基本定義、調用示例、圖像繪製、函數優化和應用實例。 一、函數基本定義 分段函數又稱為條件函數,指一條直線段或曲線段,由…

    編程 2025-04-29
  • Python函數名稱相同參數不同:多態

    Python是一門面向對象的編程語言,它強烈支持多態性 一、什麼是多態多態是面向對象三大特性中的一種,它指的是:相同的函數名稱可以有不同的實現方式。也就是說,不同的對象調用同名方法…

    編程 2025-04-29

發表回復

登錄後才能評論