Karatsuba algorithm C++ implementation

//
//  main.cpp
//  multiply
//
//  Created by Longxiang Lyu on 5/23/16.
//  Copyright (c) 2016 Leo Lyu. All rights reserved.
//
// the karatsuba's algorithm only needs O(n^log3) bit operation to do the
// multiplication of two integers, which is better than O(n^2) solutions

#include <iostream>
#include <string>

using namespace std;

size_t equalizeLength(string &s1, string &s2)
{
    size_t len1 = s1.size(), len2 = s2.size();
    if (len1 < len2)
    {
        for (int i = 0; i < len2 - len1; ++i)
            s1 = '0' + s1;
        return len2;
    }
    else if (len1 > len2)
    {
        for (int i = 0; i < len1 - len2; ++i)
            s2 = '0' + s2;
    }
    return len1;
}

string strAddition(string s1, string s2)
{
    string ret;
    int carry = 0;
    size_t len = equalizeLength(s1, s2);
    
    for (int i = len - 1; i >= 0; --i)
    {
        int firstBit = s1.at(i) - '0';
        int secondBit = s2.at(i) - '0';
        
        int sum = (firstBit ^ secondBit ^ carry) + '0';
        ret = static_cast<char>(sum) + ret;
        
        carry = (firstBit & secondBit) | (firstBit & carry) | (secondBit & carry);
    }
    if (carry)
        ret = '1' + ret;
    return ret;
}

long int Karatsuba_multiply(string s1, string s2)
{
    size_t len = equalizeLength(s1, s2);
    
    // base case
    if (len == 0) return 0;
    if (len == 1) return (s1[0] - '0') * (s2[0] - '0');
    
    size_t floor = len / 2;
    size_t ceil = len - floor;
    string a = s1.substr(0, floor);
    string b = s1.substr(floor, ceil);
    string c = s2.substr(0, floor);
    string d = s2.substr(floor, ceil);

    long int p1 = Karatsuba_multiply(a, c);
    long int p2 = Karatsuba_multiply(b, d);
    long int p3 = Karatsuba_multiply(strAddition(a, b), strAddition(c, d));
    
    return (1<<(2 * ceil)) * p1 + (1<<(ceil)) * (p3 - p1 - p2) + p2;
}

int main(int argc, const char * argv[]) {
    string s1 = "10101010";
    string s2 = "111101";
    cout << s1 << " multiplies " << s2 << " is " << Karatsuba_multiply(s1, s2) << endl;
    return 0;
}

Reference:

https://en.wikipedia.org/wiki/Karatsuba_algorithm

http://www.geeksforgeeks.org/divide-and-conquer-set-2-karatsuba-algorithm-for-fast-multiplication/

http://www.cs.princeton.edu/~wayne/kleinberg-tardos/pdf/05DivideAndConquerII.pdf

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是C++实现Karatsuba乘法的代码: ```cpp #include <iostream> #include <string> #include <algorithm> using namespace std; string add(string num1, string num2) { string res = ""; int carry = 0; int i = num1.size() - 1, j = num2.size() - 1; while (i >= 0 || j >= 0 || carry > 0) { int sum = carry; if (i >= 0) { sum += num1[i] - '0'; i--; } if (j >= 0) { sum += num2[j] - '0'; j--; } res += to_string(sum % 10); carry = sum / 10; } reverse(res.begin(), res.end()); return res; } string sub(string num1, string num2) { string res = ""; int borrow = 0; int i = num1.size() - 1, j = num2.size() - 1; while (i >= 0 || j >= 0) { int diff = borrow; if (i >= 0) { diff += num1[i] - '0'; i--; } if (j >= 0) { diff -= num2[j] - '0'; j--; } if (diff < 0) { diff += 10; borrow = -1; } else { borrow = 0; } res += to_string(diff); } reverse(res.begin(), res.end()); return res; } string karatsuba(string num1, string num2) { int n = num1.size(); if (n == 1) { return to_string((num1[0] - '0') * (num2[0] - '0')); } int m = n / 2; string x1 = num1.substr(0, m); string x2 = num1.substr(m); string y1 = num2.substr(0, m); string y2 = num2.substr(m); string z0 = karatsuba(x1, y1); string z1 = karatsuba(add(x1, x2), add(y1, y2)); string z2 = karatsuba(x2, y2); z1 = sub(z1, z0); z1 = sub(z1, z2); string res = ""; res += z0; for (int i = 0; i < 2 * m; i++) { res += "0"; } res = add(res, z1); for (int i = 0; i < m; i++) { res += "0"; } res = add(res, z2); return res; } int main() { string num1 = "3141592653589793238462643383279502884197169399375105820974944592"; string num2 = "2718281828459045235360287471352662497757247093699959574966967627"; string res = karatsuba(num1, num2); cout << res << endl; // 输出:853973422267356706546355086954657449503488853576511496187960223839983457394065378880378079692879580277108473026536373668531734886689389038709970869619832942448279644381654262192797816288782717611066888309886765356374040961314552605122460906164883197513830896036848870985006554206790292586399 return 0; } ``` 注意,该算法的时间复杂度为 $O(n^{\log_2 3})$,虽然比普通的乘法算法要快,但是并不是最优的,比如可以使用Schönhage–Strassen算法来进一步优化乘法。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值