Karatsuba 大整數(shù)乘法算法 (Nim 語言實(shí)現(xiàn))

我們平時接觸的長乘法钥星,按位相乘闯两,是一種時間復(fù)雜度為 O(n ^ 2) 的算法笔诵。今天钟病,我們來介紹一種萧恕,時間復(fù)雜度為 O (n ^ log 3) 的大整數(shù)乘法(log 表示以 2 為底的對數(shù))。

介紹原理

karatsuba 算法要求乘數(shù)與被乘數(shù)要滿足以下幾個條件肠阱,第一票唆,乘數(shù)與被乘數(shù)的位數(shù)相同;第二屹徘,乘數(shù)與被乘數(shù)的位數(shù)應(yīng)為 2 次冪走趋,即為 2 ^ 2, 2 ^ 3, 2 ^ 4, 2 ^ n 等數(shù)值噪伊。

下面我們先來看幾個簡單的例子簿煌,并以此來了解 karatsuba 算法的使用方法氮唯。

兩位數(shù)相乘

我們設(shè)被乘數(shù) A = 85,乘數(shù) B = 41姨伟。下面來看我們的操作步驟:

將 A, B 一分為二惩琉,令 p = A 的前半部分 = 8,q = A 的后半部分 = 5 夺荒, r = B 的前半部分 = 4 瞒渠,s = B 的后半部分 = 1,n = 2技扼。通過簡單的數(shù)學(xué)運(yùn)算:

A * B = pq * rs = (p * 10 + q) * (r * 10 + s) = p * r * 10 ^ 2 + (p * s + q * r ) * 10 + q * s伍玖。

令 u = p * r,v = (p - q) * (s - r)剿吻,w = q * s窍箍。 所以 A * B = u * 10 ^ 2 + (u + v + w) * 10 + w。

換成數(shù)值求解的過程如下:

A * B = 85 * 41 = (8 * 10 + 5) * ( 4 * 10 + 1) = 8 * 4 * 10 * 10 + (8 * 1 + 5 * 4) * 10 + 5 * 1和橙。其中 u = 8 * 4 = 32仔燕,v = (8 - 5) (1 - 4) = -9,w = 5 * 1 = 5魔招。所以晰搀,A * B = 32 * 100 + (32 - 9 + 5) * 10 + 5 = 3485。與長乘法所得結(jié)果一致办斑。

四位數(shù)相乘

我們設(shè)被乘數(shù) A = 8537外恕,乘數(shù) B = 4123。下面來看我們的操作步驟:

將 A, B 一分為二乡翅,令 p = A 的前半部分 = 85鳞疲,q = A 的后半部分 = 37 , r = B 的前半部分 = 41 蠕蚜,s = B 的后半部分 = 23尚洽,n = 4。

==> 其中靶累,u = 85 * 41, v = (85 - 37) * (23 - 41), w = 37 * 23腺毫。

==> A * B = 8537 * 4123 = u * 10 ^ 4 + (u + v + w) * 10 ^ 2 + w = 3485_0000 +34_7200 + 851 = 35198051。

在我們計算 u挣柬, v, w 的過程中又會涉及兩位數(shù)的乘法潮酒,我們繼續(xù)使用 Karatsuba 算法得出兩位數(shù)相乘的結(jié)果。

N 位數(shù)相乘

我們令 n 為 乘數(shù)與被乘數(shù)的位數(shù)邪蛔,令 p = A 的前半部分急黎,q = A 的后半部分, r = B 的前半部分 ,s = B 的后半部分勃教。

==> 其中淤击, u = p * r,v = (p - q) * (s - r)故源,w = q * s遭贸。 所以 A * B = u * 10 ^ n + (u + v + w) * 10 ^ (n / 2) + w。

而 u, v, w 則是兩個 n / 2 位的乘法運(yùn)算心软。我們繼續(xù)調(diào)用 Karatsuba 算法計算 u, v, w 的數(shù)值。接著著蛙,我們在計算 n / 2 乘法的過程中又會遇到 n / 4 位的乘法運(yùn)算……以此類推删铃,直到我們遇到兩個個位數(shù)的乘法,我們就直接返回這兩個個位數(shù)乘法的結(jié)果踏堡。層層返回猎唁,最終得到 N 位數(shù)的乘法結(jié)果。

代碼實(shí)現(xiàn)

代碼實(shí)現(xiàn)時顷蟆,需要將大整數(shù)先轉(zhuǎn)換為字符串诫隅,然后再截取相應(yīng)部分的數(shù)字,最后計算出最后結(jié)果帐偎。

# 關(guān)注微信公眾號:nim 編程
import math
import strutils


proc kara(a: int64, b: int64): int64 =
  if a < 10 or b < 10: 
    return a * b
  var a_str = $a 
  var b_str = $b
  var a_len = a_str.len
  var b_len = b_str.len
  let real_len = float(max(a_len, b_len))
  let max_len = 2 ^ int(ceil(log2(real_len)))
  let mid = max_len shr 1
  while a_len < max_len:
    a_str = "0" & a_str
    a_len += 1
  while b_len < max_len:
    b_str = "0" & b_str
    b_len += 1
  let p = parseInt(a_str[0 .. mid-1])
  let q = parseInt(a_str[mid .. a_str.len-1])
  let r = parseInt(b_str[0 .. mid-1])
  let s = parseInt(b_str[mid .. b_str.len-1])
  let u = kara(p, r)
  let v = kara(q - p, s - r)
  let w = kara(q, s)
  return u * 10 ^ max_len + (u + w - v) * 10 ^ mid + w

輸出結(jié)果:

==> echo kara(123456, 9734) == 123456 * 9734

==> echo kara(1234233456756, 32459734) == 1234233456756 * 32459734

時間復(fù)雜度

我們平常使用的長乘法逐纬,是 O (n ^ 2) 的時間復(fù)雜度。比如兩個 N 位數(shù)相乘削樊,我們需要將每一位按規(guī)則相乘豁生,所以需要計算 N * N 次乘法。而使用 Karatsuba 算法每層需要計算三次乘法漫贞,兩次加法甸箱,以及若干次加法,每使用一次 karatsuba 算法迅脐,乘法規(guī)模就下降一半芍殖。所以,對于兩個 n = 2 ^ K 位數(shù)乘法運(yùn)算谴蔑,我們需要計算 3 ^ K 次乘法運(yùn)算豌骏。

而 K = log n(底數(shù)為 2), 3 ^ K = 3 ^ log n = 2 ^ (log 3 * log n) = 2 ^ (log n * log 3) = n ^ log 3 (底數(shù)為 2)树碱。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末肯适,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子成榜,更是在濱河造成了極大的恐慌框舔,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,884評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異刘绣,居然都是意外死亡樱溉,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,347評論 3 385
  • 文/潘曉璐 我一進(jìn)店門纬凤,熙熙樓的掌柜王于貴愁眉苦臉地迎上來福贞,“玉大人,你說我怎么就攤上這事停士⊥诹保” “怎么了?”我有些...
    開封第一講書人閱讀 157,435評論 0 348
  • 文/不壞的土叔 我叫張陵恋技,是天一觀的道長拇舀。 經(jīng)常有香客問我,道長蜻底,這世上最難降的妖魔是什么骄崩? 我笑而不...
    開封第一講書人閱讀 56,509評論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮薄辅,結(jié)果婚禮上要拂,老公的妹妹穿的比我還像新娘。我一直安慰自己站楚,他們只是感情好脱惰,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,611評論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著窿春,像睡著了一般枪芒。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上谁尸,一...
    開封第一講書人閱讀 49,837評論 1 290
  • 那天舅踪,我揣著相機(jī)與錄音,去河邊找鬼良蛮。 笑死抽碌,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的决瞳。 我是一名探鬼主播货徙,決...
    沈念sama閱讀 38,987評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼皮胡!你這毒婦竟也來了痴颊?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,730評論 0 267
  • 序言:老撾萬榮一對情侶失蹤屡贺,失蹤者是張志新(化名)和其女友劉穎蠢棱,沒想到半個月后锌杀,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,194評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡泻仙,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,525評論 2 327
  • 正文 我和宋清朗相戀三年糕再,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片玉转。...
    茶點(diǎn)故事閱讀 38,664評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡突想,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出究抓,到底是詐尸還是另有隱情猾担,我是刑警寧澤,帶...
    沈念sama閱讀 34,334評論 4 330
  • 正文 年R本政府宣布刺下,位于F島的核電站垒探,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏怠李。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,944評論 3 313
  • 文/蒙蒙 一蛤克、第九天 我趴在偏房一處隱蔽的房頂上張望捺癞。 院中可真熱鬧,春花似錦构挤、人聲如沸髓介。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,764評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽唐础。三九已至,卻和暖如春矾飞,著一層夾襖步出監(jiān)牢的瞬間一膨,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,997評論 1 266
  • 我被黑心中介騙來泰國打工洒沦, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留豹绪,地道東北人。 一個月前我還...
    沈念sama閱讀 46,389評論 2 360
  • 正文 我出身青樓申眼,卻偏偏與公主長得像瞒津,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子括尸,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,554評論 2 349

推薦閱讀更多精彩內(nèi)容

  • 我們平時接觸的長乘法巷蚪,按位相乘,是一種時間復(fù)雜度為 O(n ^ 2) 的算法濒翻。今天屁柏,我們來介紹一種啦膜,時間復(fù)雜度為 ...
    Python高效編程閱讀 1,268評論 0 1
  • "use strict";function _classCallCheck(e,t){if(!(e instanc...
    久些閱讀 2,028評論 0 2
  • (開經(jīng)偈) 無上甚深微妙法 百千萬劫難遭遇 我今見聞得受持 愿解如來真實(shí)義 第一品 Fǎ huì yīn yóu ...
    黃一軒閱讀 3,656評論 0 1
  • 這里的詞匯不是現(xiàn)在流行的嚴(yán)格意義層面的翻譯,而是一種帶有想象力的對照前联。有元音或者輔音相同的假定的應(yīng)用功戚;有事物的聯(lián)系...
    共通語言閱讀 4,627評論 0 1
  • 謊言有無限的組合方式,但真相只有一種存在狀態(tài)似嗤。
    Rachelsrp閱讀 207評論 0 0