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

待我称王封你为后i 2023-07-13 03:17 120阅读 0赞

前一段时间,小灰发布了一篇有关大整数相加的漫画,没看过的小伙伴可以先看一看:

漫画:如何实现大整数相加?(修订版)

那么,大整数相乘又是如何实现的呢?

起初,小灰认为只要按照大整数相加的思路稍微做一下变形,就可以轻松实现大整数相乘。但是随着深入的学习,小灰才发现事情并没有那么简单……

format_png

format_png 1

————— 第二天 —————

format_png 2

format_png 3

format_png 4

format_png 5

format_png 6

怎样列出这个乘法竖式呢?以 93281 X 2034 为例,竖式如下:

format_png 7

在程序中,我们可以利用int型数组,把两个大整数按位进行存储,再把数组中的元素像小学竖式那样逐个进行计算。

这个乘法竖式的计算过程可以大体分为两步:

1.整数B的每一个数位和整数A所有数位依次相乘,得到中间结果。

2.所有中间结果相加,得到最终结果。

format_png 8

  1. /**
  2. * 大整数求乘积
  3. * @param bigNumberA 大整数A
  4. * @param bigNumberB 大整数B
  5. */
  6. public static String multiply(String bigNumberA, String bigNumberB) {
  7. //1.把两个大整数用数组逆序存储,数组长度等于两整数长度之和
  8. int lengthA = bigNumberA.length();
  9. int lengthB = bigNumberB.length();
  10. int[] arrayA = new int[lengthA];
  11. for(int i=0; i< lengthA; i++){
  12. arrayA[i] = bigNumberA.charAt(lengthA-1-i) - '0';
  13. }
  14. int[] arrayB = new int[lengthB];
  15. for(int i=0; i< lengthB; i++){
  16. arrayB[i] = bigNumberB.charAt(lengthB-1-i) - '0';
  17. }
  18. //2.构建result数组,数组长度等于两整数长度之和
  19. int[] result = new int[lengthA+lengthB];
  20. //3.嵌套循环,整数B的每一位依次和整数A的所有数位相乘,并把结果累加
  21. for(int i=0;i<lengthB;i++) {
  22. for(int j=0;j<lengthA;j++) {
  23. //整数B的某一位和整数A的某一位相乘
  24. result[i+j] += arrayB[i]*arrayA[j];
  25. //如果result某一位大于10,则进位,进位数量是该位除以10的商
  26. if(result[i+j] >= 10){
  27. result[i+j+1] += result[i+j]/10;
  28. result[i+j] = result[i+j]%10;
  29. }
  30. }
  31. }
  32. //4.把result数组再次逆序并转成String
  33. StringBuilder sb = new StringBuilder();
  34. //是否找到大整数的最高有效位
  35. boolean findFirst = false;
  36. for (int i = result.length - 1; i >= 0; i--) {
  37. if(!findFirst){
  38. if(result[i] == 0){
  39. continue;
  40. }
  41. findFirst = true;
  42. }
  43. sb.append(result[i]);
  44. }
  45. return sb.toString();
  46. }
  47. public static void main(String[] args) {
  48. String x = "3338429042340042304302404";
  49. String y = "12303231";
  50. System.out.println(multiply(x, y));
  51. }

format_png 9

format_png 10

format_png 11

format_png 12

format_png 13

format_png 14

————————————

format_png 15

format_png 16

format_png 17

format_png 18

format_png 19

format_png 20

format_png 21

下面,我们的推导会有一些烧脑,请大家坐稳扶好~~

大整数从高位到低位,被平分成了两部分。设整数1的高位部分是A,低位部分是B;整数2的高位部分是C,低位部分是D,那么有如下等式:

format_png 22

如果把大整数的长度抽象为n,那么:

format_png 23

因此,整数1与整数2 的乘积可以写成下面的形式:

format_png 24

如此一来,原本长度为**n的大整数的1次乘积,被转化成了长度为n/2的大整数的4次**乘积(AC,AD,BC,BD)。

format_png 25

format_png 26

format_png 27

format_png 28

format_png 29

什么是master定理呢?

master定理的英语名称是master theorem,它为许多由分治法得到的递推关系式提供了渐进时间复杂度分析。

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

format_png 30

format_png 31

format_png 32

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

根据刚才得到的结论,两个大整数相乘被拆分成四个较小的乘积:

format_png 24

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

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

其中f(n)是4个乘积结果相加的运算规模,f(n)的渐进时间复杂度很明显是O(n)

把这个关系带入到master定理的公式 T(n) = a T(n / b) + f(n) 当中,

此时 a=4, b=2

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

format_png 33

发现正好符合条件。

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

format_png 34

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

format_png 35

format_png 36

format_png 37

format_png 38

format_png 39

format_png 40

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

format_png 41

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

format_png 42

format_png 43

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

假设两个长度为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),那么则有如下规律:

format_png 30

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

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

format_png 33

发现正好符合条件。

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

format_png 44

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

format_png 45

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

format_png 46

format_png 47

format_png 48

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

  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. }

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

format_png 49

format_png 50

format_png 51

format_png 52

format_png 53

format_png 54

几点补充:

  1. 文章最后的代码,经由网上技术博客的代码改动而来,仅做参考。

  2. 关于快速傅里叶变换,有兴趣深入研究的小伙伴们可以参考《算法导论》第30章的内容。

发表评论

表情:
评论列表 (有 0 条评论,120人围观)

还没有评论,来说两句吧...

相关阅读