漫画:如何实现大整数相乘?(下)

戳蓝字“CSDN云计算”关注我们哦!


如何用程序实现大整数相乘呢?


在上一篇文章  漫画:如何实现大整数相乘?(上) 当中,我们介绍了两种思路:


1.像列竖式一样,把两整数按位依次相乘

640?wx_fmt=png


这个思路的时间复杂度是O(n^2)。



2.利用分治法,把每个大整数分成高位和低位两部分,转化成四个较小的乘积。


640?wx_fmt=png

这个思路的时间复杂度同样是O(n^2)。


那么,有什么样的优化方案,可以使时间复杂度优于O(n^2)呢?我们今天一起来研究下。



640?wx_fmt=jpeg



640?wx_fmt=jpeg


640?wx_fmt=jpeg


640?wx_fmt=jpeg



如何做调整呢?其实很简单,连小学生都会:


640?wx_fmt=png


这样一来,原本的4次乘法和3次加法,转变成了3次乘法和6次加法


640?wx_fmt=jpeg


640?wx_fmt=jpeg



这样一来,时间复杂度是多少呢?


假设两个长度为n的大整数相乘,整体运算规模是T(n) 。


刚才我们说过,两个大整数相乘可以被拆分成三个较小的乘积,

所以在第一次分治时,T(n)和T(n/2)有如下关系:

T(n) = 3T(n/2) + f(n)

其中f(n)是6次加法的运算规模,f(n)的渐进时间复杂度很明显是O(n)


此时让我们回顾一下master定理:


设常数a >= 1,b > 1,如果一个算法的整体计算规模 T(n) =  a T(n / b) + f(n),那么则有如下规律:


640?wx_fmt=png

对于T(n) = 3T(n/2) + f(n)这个关系式来说, a=3, b=2


把a和b的值,以及f(n)的时间复杂度带入到master定理的第一个规律,也就是下面的规律:


640?wx_fmt=png


发现正好符合条件。


怎么符合条件呢?推导过程如下:


640?wx_fmt=png


所以我们的平均时间复杂度是:

640?wx_fmt=png


2 和 1.59 之间的差距看似不大,但是当整数长度非常大的时候,两种方法的性能将是天壤之别。


640?wx_fmt=jpeg


640?wx_fmt=jpeg


640?wx_fmt=png


下面展示一下实现代码。我们的代码非常复杂,在这里只作为参考,最重要的还是解决问题的思路:


  1. /**

  2. * 大整数乘法

  3. * @param bigNumberA  大整数A

  4. * @param bigNumberB  大整数B

  5. */

  6. public static String bigNumberMultiply(String bigNumberA, String bigNumberB) {

  7.    boolean isNegative = false;

  8.    if ((bigNumberA.startsWith("-") && bigNumberB.startsWith("-"))

  9.            || (!bigNumberA.startsWith("-") && !bigNumberB.startsWith("-"))) {

  10.        // 两数同符号的情况

  11.        bigNumberA = bigNumberA.replaceAll("-", "");

  12.        bigNumberB = bigNumberB.replaceAll("-", "");

  13.    } else if ((bigNumberA.startsWith("-") && !bigNumberB.startsWith("-"))

  14.            || (!bigNumberA.startsWith("-") && bigNumberB.startsWith("-"))) {

  15.        // 两数不同符号的情况

  16.        bigNumberA = bigNumberA.replace("-", "");

  17.        bigNumberB = bigNumberB.replace("-", "");

  18.        isNegative = true;

  19.    }

  20.    // 如果两数长度之和小于10,直接相乘返回

  21.    if (bigNumberA.length() + bigNumberB.length() < 10) {

  22.        // 计算乘积

  23.        int tmp = (Integer.parseInt(bigNumberA) * Integer.parseInt(bigNumberB));

  24.        if (tmp == 0) {

  25.            return "0";

  26.        }

  27.        String value = String.valueOf(tmp);

  28.        if(isNegative){

  29.            value = "-" + value;

  30.        }

  31.        return value;

  32.    }

  33.    // 公式 AC * 10^n+((A-B)(D-C)+AC+BD) * 10^(n/2)+BD当中的a,b,c,d

  34.    String a, b, c, d;

  35.    if (bigNumberA.length() == 1) {

  36.        a = "0";

  37.        b = bigNumberA;

  38.    } else {

  39.        if (bigNumberA.length() % 2 != 0) {

  40.            bigNumberA = "0" + bigNumberA;

  41.        }

  42.        a = bigNumberA.substring(0, bigNumberA.length() / 2);

  43.        b = bigNumberA.substring(bigNumberA.length() / 2);

  44.    }

  45.    if (bigNumberB.length() == 1) {

  46.        c = "0";

  47.        d = bigNumberB;

  48.    } else {

  49.        if (bigNumberB.length() % 2 != 0) {

  50.            bigNumberB = "0" + bigNumberB;

  51.        }

  52.        c = bigNumberB.substring(0, bigNumberB.length() / 2);

  53.        d = bigNumberB.substring(bigNumberB.length() / 2);

  54.    }

  55.    // 按最大位数取值,以确定补零数目

  56.    int n = bigNumberA.length() >= bigNumberB.length() ? bigNumberA.length() : bigNumberB.length();


  57.    //t1,t2为中间运算结果,t3为乘法运算完毕的结果

  58.    String t1, t2, t3;

  59.    String ac = bigNumberMultiply(a, c);

  60.    String bd = bigNumberMultiply(b, d);


  61.    //t1=(A-B)(D-C)

  62.    t1 = bigNumberMultiply(bigNumberSubtract(a, b), bigNumberSubtract(d, c));

  63.    //t2=(A-B)(D-C)+AC+BD

  64.    t2 = bigNumberSum(bigNumberSum(t1, ac), bd);

  65.    //t3= AC * 10^n+((A-B)(D-C)+AC+BD) * 10^(n/2)+BD

  66.    t3 = bigNumberSum(bigNumberSum(Power10(ac, n), Power10(t2, n/2)), bd).replaceAll("^0+", "");

  67.    if (t3 == "")

  68.        return "0";

  69.    if(isNegative){

  70.        return "-" + t3;

  71.    }

  72.    return t3;

  73. }



  74. /**

  75. * 大整数加法

  76. * @param bigNumberA  大整数A

  77. * @param bigNumberB  大整数B

  78. */

  79. public static String bigNumberSum(String bigNumberA, String bigNumberB) {


  80.    if (bigNumberA.startsWith("-") && !bigNumberB.startsWith("-")) {

  81.        return bigNumberSubtract(bigNumberB, bigNumberA.replaceAll("^-", ""));

  82.    } else if (!bigNumberA.startsWith("-") && bigNumberB.startsWith("-")) {

  83.        return bigNumberSubtract(bigNumberA, bigNumberB.replaceAll("^-", ""));

  84.    } else if (bigNumberA.startsWith("-") && bigNumberB.startsWith("-")) {

  85.        return "-" + bigNumberSum(bigNumberA.replaceAll("^-", ""), bigNumberB.replaceAll("^-", ""));

  86.    }


  87.    //1.把两个大整数用数组逆序存储,数组长度等于较大整数位数+1

  88.    int maxLength = bigNumberA.length() > bigNumberB.length() ? bigNumberA.length() : bigNumberB.length();

  89.    int[] arrayA = new int[maxLength+1];

  90.    for(int i=0; i< bigNumberA.length(); i++){

  91.        arrayA[i] = bigNumberA.charAt(bigNumberA.length()-1-i) - '0';

  92.    }

  93.    int[] arrayB = new int[maxLength+1];

  94.    for(int i=0; i< bigNumberB.length(); i++){

  95.        arrayB[i] = bigNumberB.charAt(bigNumberB.length()-1-i) - '0';

  96.    }

  97.    //2.构建result数组,数组长度等于较大整数位数+1

  98.    int[] result = new int[maxLength+1];

  99.    //3.遍历数组,按位相加

  100.    for(int i=0; i<result.length; i++){

  101.        int temp = result[i];

  102.        temp += arrayA[i];

  103.        temp += arrayB[i];

  104.        //判断是否进位

  105.        if(temp >= 10){

  106.            temp -= 10;

  107.            result[i+1] = 1;

  108.        }

  109.        result[i] = temp;

  110.    }

  111.    //4.把result数组再次逆序并转成String

  112.    StringBuilder sb = new StringBuilder();

  113.    //是否找到大整数的最高有效位

  114.    boolean findFirst = false;

  115.    for (int i = result.length - 1; i >= 0; i--) {

  116.        if(!findFirst){

  117.            if(result[i] == 0){

  118.                continue;

  119.            }

  120.            findFirst = true;

  121.        }

  122.        sb.append(result[i]);

  123.    }

  124.    return sb.toString();

  125. }


  126. /**

  127. * 大整数减法

  128. * @param bigNumberA  大整数A

  129. * @param bigNumberB  大整数B

  130. */

  131. public static String bigNumberSubtract(String bigNumberA, String bigNumberB) {

  132.    int compareResult = compare(bigNumberA, bigNumberB);

  133.    if (compareResult == 0) {

  134.        return "0";

  135.    }

  136.    boolean isNegative = false;

  137.    if (compareResult == -1) {

  138.        String tmp = bigNumberB;

  139.        bigNumberB = bigNumberA;

  140.        bigNumberA = tmp;

  141.        isNegative = true;

  142.    }

  143.    //1.把两个大整数用数组逆序存储,数组长度等于较大整数位数+1

  144.    int maxLength = bigNumberA.length() > bigNumberB.length() ? bigNumberA.length() : bigNumberB.length();

  145.    int[] arrayA = new int[maxLength+1];

  146.    for(int i=0; i< bigNumberA.length(); i++){

  147.        arrayA[i] = bigNumberA.charAt(bigNumberA.length()-1-i) - '0';

  148.    }

  149.    int[] arrayB = new int[maxLength+1];

  150.    for(int i=0; i< bigNumberB.length(); i++){

  151.        arrayB[i] = bigNumberB.charAt(bigNumberB.length()-1-i) - '0';

  152.    }

  153.    //2.构建result数组,数组长度等于较大整数位数+1

  154.    int[] result = new int[maxLength+1];

  155.    //3.遍历数组,按位相加

  156.    for(int i=0; i<result.length; i++){

  157.        int temp = result[i];

  158.        temp += arrayA[i];

  159.        temp -= arrayB[i];

  160.        //判断是否进位

  161.        if(temp < 0){

  162.            temp += 10;

  163.            result[i+1] = -1;

  164.        }

  165.        result[i] = temp;

  166.    }

  167.    //4.把result数组再次逆序并转成String

  168.    StringBuilder sb = new StringBuilder();

  169.    //是否找到大整数的最高有效位

  170.    boolean findFirst = false;

  171.    for (int i = result.length - 1; i >= 0; i--) {

  172.        if(!findFirst){

  173.            if(result[i] == 0){

  174.                continue;

  175.            }

  176.            findFirst = true;

  177.        }

  178.        sb.append(result[i]);

  179.    }

  180.    String value = sb.toString();

  181.    if (isNegative) {

  182.        value = "-" + value;

  183.    }

  184.    return value;

  185. }


  186. // 比较大小

  187. private static int compare(String x, String y) {

  188.    if (x.length() > y.length()) {

  189.        return 1;

  190.    } else if (x.length() < y.length()) {

  191.        return -1;

  192.    } else {

  193.        for (int i = 0; i < x.length(); i++) {

  194.            if (x.charAt(i) > y.charAt(i)) {

  195.                return 1;

  196.            } else if (x.charAt(i) < y.charAt(i)) {

  197.                return -1;

  198.            }

  199.        }

  200.        return 0;

  201.    }

  202. }


  203. // 扩大10的n次方倍

  204. public static String Power10(String num, int n) {

  205.    for (int i = 0; i < n; i++) {

  206.        num += "0";

  207.    }

  208.    return num;

  209. }


  210. public static void main(String[] args) {

  211.    String x = "1513143";

  212.    String y = "9345963";

  213.    System.out.println(bigNumberMultiply(x, y));

  214. }


需要注意的是,这段实现代码只适用于两个大整数长度相等的情况。如果想求解长度不等的整数相乘,只需要对代码做微小的改动,有兴趣的小伙伴没有试一试。


640?wx_fmt=jpeg


640?wx_fmt=jpeg


640?wx_fmt=jpeg


640?wx_fmt=jpeg


640?wx_fmt=jpeg



640?wx_fmt=jpeg


文章转程序员小灰


1.微信群:

添加小编微信:color_ld,备注“进群+姓名+公司职位”即可,加入【云计算学习交流群】,和志同道合的朋友们共同打卡学习!


2.征稿:

投稿邮箱:liudan@csdn.net;微信号:color_ld。请备注投稿+姓名+公司职位。


推荐阅读

  • 下一次 IT 变革:边缘计算(Edge computing)

  • 为什么 ofo 彻底凉了?| 畅言

  • AI in 美团:吃喝玩乐背后的黑科技

  • 无业务不技术:那些誓用区块链重塑的行业,发展怎么样了?

  • Windows 成“弃子”,Linux 终上位?

  • 突发!12306 脱库 410 万用户数据究竟从何泄漏?

  • 可替代Android的6大开源移动操作系统

  • 程序员求助:被领导强行要求写Bug该怎么办?网友的回答让我笑翻


程序员抢票姿势 ↓交朋友还能抢票?


640?wx_fmt=jpeg

为交流学习,请备注+姓名+公司职位(学校专业)


640?wx_fmt=gif点击“阅读原文”,打开 CSDN App 阅读更贴心!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/525035.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

BugkuCTF-MISC题宽带信息泄露

下载文件conf.zip&#xff0c;解压得出conf.bin 在端输入strings conf.bin没有flag 此题需要路由器配置文件查看工具RouterPassView 下载此工具 在终端输入wine RouterPassView.exe(wine即在linux系统里打开exe程序文件) 查找username 提交flag{053700357621}成功

全世界的程序员们,为什么都不在意“穿衣”这档事?

戳蓝字“CSDN云计算”关注我们哦&#xff01;十月份我去北京出差&#xff0c;来到后厂村走了一圈&#xff0c;想看看现实中的北京程序员是否真的像之前刷屏的《互联网公司时尚穿搭指南》中所述&#xff0c;存在“天然结界”。到的时候正好是午休时间&#xff0c;一下车就在某巨…

BugkuCTF-MISC题猫片

python知识点 ^是按位异或操作符 [::-1]能令列表反向 解题流程 题目提示&#xff1a;LSB BGR NTFS 下载之后就是一个叫png的文件&#xff0c;没有后缀&#xff0c;那么加后缀 打开Stegsolve直接就Analyse-----Data Extract&#xff0c;根据提示LSB,BGR,一顿尝试 发现了PN…

等了20年的物联网,这次真的会成功吗?

戳蓝字“CSDN云计算”关注我们哦&#xff01;最近几年&#xff0c;物联网的概念非常火爆&#xff0c;和物联网相关的技术&#xff0c;例如NB-IoT、LoRa、eMTC等&#xff0c;都频繁地出现在我们的眼前&#xff0c;出现在新闻里&#xff0c;出现在展会上。其实&#xff0c;「物联…

BugkuCTF-MISC题妹子的陌陌

下载图片 通过010editor查找并未发现有价值的信息 通过binwalk -e file.jpg分离出文件夹_file.jpg.extracted 里有91DC.rar&#xff0c;解压发现需要密码&#xff0c;通过爆破找不到密码 在file.jpg图片里有“喜欢我吗.”&#xff0c;将其输入到解压密码里 发现竟成功解压 得到…

深入理解与应用Hadoop中的MapReduce

戳蓝字“CSDN云计算”关注我们哦&#xff01;作者&#xff1a;安静的技术控 来源&#xff1a;CSDN 原文&#xff1a;https://blog.csdn.net/a2011480169/article/details/52924955 版权声明&#xff1a;本文为博主原创文章&#xff0c;转载请附上博文链接&#xff01;现在大数据…

BugkuCTF-MISC题蜜雪冰城~

看txt&#xff0c;里面只有(0, 0, 0)和(255, 255, 255)&#xff0c;(0, 0, 0)是黑色&#xff0c;替换成1&#xff0c;(255, 255, 255)是白色&#xff0c;替换成0。 脚本&#xff1a; with open("你爱我呀我爱你.txt",) as f:with open("binary.txt",w) as…

你好,请查收这封元旦祝福,CSDN云计算改版啦!

戳蓝字“CSDN云计算”关注我们哦&#xff01;◆ ◆ ◆ ◆2019新年快乐HAPPY NEW YEAR此时是2019年1月1日站在新一年的起跑线上最暖心的鼓励让2019更精彩祝福中&#xff0c;你步伐坚定&#xff0c;走向远方 ◆ ◆ ◆ ◆元旦快乐CSDN云计算的粉丝们&#xff1a;好消息&#xff01…

C语言 数组定义和使用 - C语言零基础入门教程

目录 一.数组简介二.定义数组并初始化 1.定义数组2.初始化数组 三.访问数组四.修改数组五.猜你喜欢 零基础 C/C 学习路线推荐 : C/C 学习目录 >> C 语言基础入门 一.数组简介 在 C 语言中&#xff0c;一组数据的集合称为数组 Array &#xff0c;其实前面我们已经学习过…

BugkuCTF-MISC题普通的二维码

考点在八进制转字符串 下载文件file.zip解压得到一张二维码&#xff0c;扫描没有flag 通过010editor打开&#xff0c;在尾部有一串数字 字符串为&#xff1a; 14615414114717311014116614513717106012513712017113716314316215116016413711716414313712415713712414515613710…

C语言 数组遍历 - C语言零基础入门教程

目录 一.数组遍历原理二.数组遍历实战 1.遍历数组查询数组中的元素2.遍历数组修改数组中的元素 三.猜你喜欢 零基础 C/C 学习路线推荐 : C/C 学习目录 >> C 语言基础入门 一.数组遍历原理 在 C / C 数组定义和初始化中详细的介绍了关于数组五种初始化方法&#xff0c;这…

2018 疯狂微服务之死

戳蓝字“CSDN云计算”关注我们哦&#xff01;近期微服务的话题非常火爆&#xff0c;有时可谓非常“疯狂”&#xff1a;Netflix 在 devops 上做得很棒&#xff0c;同时 Netfix 也采用微服务。因此&#xff1a;如果我也用微服务&#xff0c;那么我也可以在 devops 方面做得很好。…

BugkuCTF-MISC题怀疑人生

补充&#xff1a;掩码爆破 字掩码爆破 所以选择数字&#xff0c;打三个英文问号&#xff0c;如果知道几位比如知道掩码是meimei后面是三个但是不知道&#xff0c;就写meimei??? 解题流程 下载文件file.zip 解压出三个文件 ctf1.zip&#xff0c;ctf2.zip&#xff0c;ctf3.…

有问有答 | 分布式服务框架精华问答

戳蓝字“CSDN云计算”关注我们哦&#xff01;随着社会的发展&#xff0c;技术的进步&#xff0c;以前的大型机架构由于高成本、难维护等原因渐渐地变得不再主流&#xff0c;替代它的就是当下最火的分布式架构。要说分布式架构有多重要&#xff0c;可以说如果你不了解“分布式”…

BugkuCTF-MISC题赛博朋克

下载文件file.zip&#xff0c;打开发现出错&#xff0c;查看是否为伪加密&#xff0c;以010editor打开搜索504B0102 处起第9&#xff0c;10个字节为0000的为伪加密&#xff0c;若改后仍不能打开为真加密 改为0000后可以打开&#xff0c;解压后得到一个文件cyberpunk.txt&#…

云要闻 | 腾讯在华南建云计算基地;致敬Larry Roberts;华云数据在无锡拿了一个奖!...

戳蓝字“CSDN云计算”关注我们哦&#xff01;Hello&#xff0c;everyone&#xff1a;1月2日&#xff0c;星期三CSDN云要闻时间&#xff1a;嗨&#xff0c;大家好&#xff0c;偶是“CSDN 云计算”微信公众号新晋成员一枚&#xff0c;人称“要闻君”。2019年刚一露头儿&#xff0…

BugkuCTF-MISC题三色绘恋

打开压缩包出错 通过winrar可解压出图片 下载文件后是个压缩包&#xff0c;要密码&#xff0c;但是没有任何信息&#xff0c;怀疑是伪加密。010打开 0900改为0000取消加密 后可以解压&#xff0c;得到一张图片。 拿到jpg图片binwalk看下发现zip文件foremost分离一下&#xf…

有问有答 | Storm技术内幕与实践精华问答

戳蓝字“CSDN云计算”关注我们哦&#xff01;Storm是Twitter开源的分布式实时大数据处理框架&#xff0c;被业界称为实时版Hadoop。 随着越来越多的场景对Hadoop的MapReduce高延迟无法容忍&#xff0c;比如网站统计、推荐系统、预警系统、金融系统等&#xff0c; 大数据实时处理…

BugkuCTF-MISC题闪的好快

下载文件&#xff0c;得到一个gif 打开&#xff0c;发现真的闪的好快啊 gif分解网站&#xff1a;https://tu.sioe.cn/gj/fenjie/ GIF动态图片分解 逐个截屏下来 再逐个通过CQR.exe扫描&#xff0c;得到 SYC{F1aSh_so_f4sT}

OpenStack 2018 年终盘点

戳蓝字“CSDN云计算”关注我们哦&#xff01;目录前言OpenStack 一年来的成长NovaCinderNeutronIronicCyborgOctaviaKollaMagnumZunKuryr从 OpenStack 到 OpenInfra前言这是最好的时代&#xff0c;也是最坏的时代 — — 查尔斯约翰赫芬姆狄更斯《双城记》年关将至&#xff0c;闲…