看了就会的大整数乘法运算与分治算法

在数据加密处理中有很多复杂的加密算法,这些加密算法往往会用到很多超大的整数运算。不过,程序设计语言对数据的大小会有一定的限制,数据太大就会出现数据溢出的情况,这是无法进行大整型数据运算的。本文将和大家一起学习如何实现大整数的数据运算,本文代码我们使用C++实现。

普通乘数运算

对于乘数运算有一种比较简单较为容易理解的方法,我们可以利用小学时期学的列竖式的计算方法进行乘法运算。

列竖式

参考上图中的列竖式计算方法,我们进行代码实现。

#include <iostream>
#include <string>
#include <stdlib.h>
#include <vector>
#include <cstring>
#include <algorithm>

std::string multiply(std::string a, std::string b)
{
    std::string result = "";
    int row = b.size();
    int col = a.size() + 1;
    int tmp[row][col];
    memset(tmp,0, sizeof(int)*row*col);
    
    reverse(a.begin(),a.end());
    reverse(b.begin(),b.end());
    
    for(int i = 0; i < b.size(); i++)
    {
        for(int j = 0; j < a.size(); j++)
        {
            std::string bit_a = std::string(1, a.at(j));
            std::string bit_b = std::string(1, b.at(i));
            
            tmp[i][j] += std::stoi(bit_a) * std::stoi(bit_b);
        
            tmp[i][j+1] = tmp[i][j] / 10;
            tmp[i][j] %= 10;

        }

    }

    int N =  a.size() + b.size();
    int sum[N];
    memset(sum, 0, sizeof(int)*N);
    
    for(int n = 0; n < N; n++)
    {
        int i = 0;
        int j = n;
        
        while (i <= n && j >= 0 )
        {
            if(i < row && j < col)
            {
                sum[n] += tmp[i][j];
            }
            
            i++;
            j--;
        }

        if( n+1 < N )
        {
            sum[n+1] = sum[n] / 10;
            sum[n] %= 10;
        }

    }

    bool zeroStartFlag = true;
    for (int i = N-1; i >= 0; i--)
    {
        if(sum[i]==0 && zeroStartFlag)
        {
            continue;
        }
        
        zeroStartFlag = false;
        result.append(std::to_string(sum[i]));
    }
    
    return result;
}


int main()
{
    std::string a = "3456";
    std::string b = "1234";

    std::string result = multiply(a, b);    
    std::cout << a << " * " << b << " = " << result <<std::endl;
    
    return 0;
}

为了方便我们先将各个乘数做了翻转处理,最后需要再将结果翻转回来。在运算过程中用来存放乘法运算结果的数组,我们没有进行移位处理同列相加,而是对角线相加,这样可以减少空间和运算步骤。上面的代码运行结果如下所示。

image

因为字符串的长度没有特别的限制,所以上面的算法可以适用大整数运算。

分治算法

虽然上面的列竖式的方法可以很好的解决大整数乘法的问题,但是我们还用一种更加高效的方法可以选择,这就是分治(Divide and Conquer)算法。它是一种非常重要的算法,可以应用到很多领域,比如快速排序,二分搜索等。从算法的名字我们可以看出它是将大的问题拆分进行细化,由大变小,先将小的问题解决,最终将这个问题解决,所以叫Divide and Conquer。在这里我们可以通过这种方法将大整数进行拆分,拆分成一个个可以通过程序语言直接进行运算的小整进行计算,最后求得到大整数的值。

假设有两个大整数,我们设为a(大小为n位)和b(大小为m位),这里我们将使用二分法对数据进行拆分,这两个整数我们可以分解为:

a = a_1 * 10^{n/2} + a_0
b = b_1 * 10^{m/2} + b_0

则,

a*b = (a_1 10^{n/2} + a_0) * (b_1 10^{m/2} + b_0)
= a_1b_1 10^{n/2 + m/2} + a_1b_0 10^{n/2} + a_0b_1 10^{m/2} + a_0b_0

令,
z_3 = a_1b_1
z_2 = a_1b_0
z_1 = a_0b_1
z_0 = a_0b_0

根据上面公式里,我们可以将a*b分解为四个小的整数的乘法,即z3,z2,z1,z0四个表达式。如果分解出来的乘法数值还是很大,还可以按照同样的方法进行拆解直到拆解成较小的数值乘法,直到计算机程序语言可以直接运算。

比如,上面的3456和1234相乘,参考下图通过二分法逐级对整数进行拆分,我们将两个整数拆分到一位整数进行运算。

image

下面我们看一下分治算法的代码实现,这里我们使用递归的方法。

#include <iostream>
#include <string>
#include <stdlib.h>
#include <vector>
#include <cstring>
#include <algorithm>
#include <cmath>

std::string add(std::string a, std::string b)
{
    int N = std::max(a.size(), b.size());
    int sum[N];
    memset(sum, 0, sizeof(int)*N);
    
    reverse(a.begin(),a.end());
    reverse(b.begin(),b.end());

    for(int i = 0; i< N; i++)
    {
        int bit_a = 0;
        int bit_b = 0;
        if(i < a.size()) bit_a = std::stoi(std::string(1, a.at(i)));
        if(i < b.size()) bit_b = std::stoi(std::string(1, b.at(i)));

        sum[i] += (bit_a + bit_b);

        if(i < N-1 && sum[i]>9)
        {
            sum[i+1] = sum[i] / 10;
            sum[i] %=10;
        }
    }

    std::string result="";
    bool zeroStartFlag = true;
    for (int i = N-1; i >= 0; i--)
    {
        if(sum[i]==0 && zeroStartFlag)
        {
            continue;
        }
        
        zeroStartFlag = false;
        result.append(std::to_string(sum[i]));
    }


    return result;
}

std::string divideAndConquer(std::string a, std::string b)
{
    if( a.size() < 2 && b.size() < 2) 
    {
        return std::to_string(std::stoi(a) * std::stoi(b));
    }
    
    int n = a.size();
    int m = b.size();
    
    int halfN = n/2;
    int halfM = m/2;

    std::string a0 = "0";
    std::string a1 = "0";
    if(a.size() > halfN && halfN > 0)
    {
        a1 = a.substr(0, halfN);
        a0 = a.substr(halfN, a.size() - halfN);
    }
    else
    {
        a1 = "0";
        a0 = a;
    }
    
    std::string b0 = "0";
    std::string b1 = "0";
    if(b.size() > halfM && halfM > 0)
    {
        b1 = b.substr(0, halfM);
        b0 = b.substr(halfM, b.size() - halfM);

    }
    else
    {
        b1 = "0";
        b0 = b;
    }

    std::string a1b1 = divideAndConquer(a1, b1);
    std::string a0b0 = divideAndConquer(a0, b0);
    std::string a1b0 = divideAndConquer(a1, b0);
    std::string a0b1 = divideAndConquer(a0, b1);
    
    a1b1.append((n - halfN) + (m - halfM), '0');
    a1b0.append(n - halfN, '0');
    a0b1.append(m - halfM, '0');

    std::string result = "";
    result = add(a1b1, a1b0);
    result = add(result, a0b1);
    result = add(result, a0b0);

    return result;
}

int main()
{
    std::string a = "3456";
    std::string b = "1234";

    std::cout << a << " * " << b << " = " << divideAndConquer(a, b) << std::endl; 

    return 0;
}


程序的运行结果如下:

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

推荐阅读更多精彩内容