整數性質(拓展歐幾裏得算法)
整數性質
時間限制:500 ms | 內存限制:65535 KB 難度:1- 描述
-
我們知道,在數學中,對於任意兩個正整數a和b,必定存在一對整數s、t使得sa+tb=gcd(a,b)。
- 輸入
- 多組測試數據。
每組數據輸入兩個非負整數a和b且a+b>0且a不等於b。
其中0<=a,b<100000。 - 輸出
- 輸出滿足條件的 s 和 t 。
- 樣例輸入
-
2 4 3 8 737 635
- 樣例輸出
-
1 0 3 -1 193 -224
1 /* 2 擴展歐幾裏德定理 3 4 對於與不完全為 0 的非負整數 a,b,gcd(a,b)表示 a,b 的最大公約數。那麽存在唯一的整 5 數 x,y 使得 gcd(a,b)=ax+by。
解題思路:(轉載http://www.cnblogs.com/zyf0163/p/4792953.html)
相信大家對歐幾裏得算法,即輾轉相除法不陌生吧。
代碼如下:
int
gcd(
int
a,
int
b){
return
!b ? gcd(b, a % b) : a;
}
而擴展歐幾裏得算法,顧名思義就是對歐幾裏得算法的擴展。
切入正題:
首先我們來看一個問題:
求整數x, y使得ax + by = 1, 如果gcd(a, b) != 1, 我們很容易發現原方程是無解的。則方程ax + by = 1有正整數對解(x, y)的必要條件是gcd(a, b) = 1,即a, b 互質。
此時正整數對解(x, y)可以通過擴展歐幾裏得算法求得。
對於方程ax + by = gcd(a, b);我們設解為x1, y1
我們令a = b, b = a % b;
得到方程bx + a % by = gcd(b, a % b);
由歐幾裏得算法可以得到gcd(a, b) = gcd(b, a % b);
代入可得:bx + a % b y = gcd(a, b)
設此方程解為x2, y2;
在計算機中我們知道: a % b = a - (a / b) * b;
代入方程化解得:
ay2 + b(x2 - (a / b) y2) = gcd(a, b);
與ax1 + by1 = gcd(a, b) 聯立,我們很容易得:
x1 = y2, y1 = x2 - (a / b)y2;
然後我們就這樣可以解出來了。
等等我們似乎忘記一個東西了吧?對就是遞歸的終點。也就是最後方程的解x和y。
對於方程ay2 + b(x2 - (a / b) y2) = gcd(a, b);
當b = 0時,發現a * 1 + b * 0 = gcd(a, b)
則有x = 1, y = 0。
由此我們把ax + by = 1的其中一組解解出來了, 僅僅是其中一組解。
對於已經得到的解x1, y1;我們便可以求出通解。
我們設x = x1 + kt;t為整數
帶入方程解得y = y1 - a * k / b * t;
而我們要保證y也為整數的話必須保證a * k /b也為整數,我們不妨令k = b/gcd(a, b);
所以通解為:
x = x1 + b / gcd(a, b) * t;
y = y1 - a / gcd(a, b) * t;
其中t為整數。
附上偽代碼:
int
a, b, x, y;
int
extgcd(
int
a,
int
b,
int
&x,
int
&y){
int
d = a;
if
(b != 0){
d = extgcd(b, a % b, y, x);
y -= (a / b) * x;
}
else
x = 1, y = 0;
return
d;
}
//d = gcd(a, b);
整數性質(拓展歐幾裏得算法)