Python 实现大整数乘法算法的示例代码


Posted in Python onSeptember 17, 2019

我们平时接触的长乘法,按位相乘,是一种时间复杂度为 O(n ^ 2) 的算法。今天,我们来介绍一种时间复杂度为 O (n ^ log 3) 的大整数乘法(log 表示以 2 为底的对数)。

介绍原理

karatsuba 算法要求乘数与被乘数要满足以下几个条件,第一,乘数与被乘数的位数相同;第二,乘数与被乘数的位数应为  2 次幂,即为 2 ^ 2,  2 ^ 3, 2 ^ 4, 2 ^ n 等数值。

下面我们先来看几个简单的例子,并以此来了解 karatsuba 算法的使用方法。

两位数相乘

我们设被乘数 A = 85,乘数 B = 41。下面来看我们的操作步骤:

将 A, B 一分为二,令 p = A 的前半部分 = 8,q = A 的后半部分 = 5 , r = B 的前半部分 = 4 ,s = B 的后半部分 =  1,n = 2。通过简单的数学运算:

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。

换成数值求解的过程如下:

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。与长乘法所得结果一致。

四位数相乘

我们设被乘数 A = 8537,乘数 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 的过程中又会涉及两位数的乘法,我们继续使用 Karatsuba 算法得出两位数相乘的结果。

N 位数相乘

我们令 n 为 乘数与被乘数的位数,令 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 位的乘法运算。我们继续调用 Karatsuba 算法计算 u, v, w 的数值。接着,我们在计算 n / 2 乘法的过程中又会遇到 n / 4 位的乘法运算……以此类推,直到我们遇到两个个位数的乘法,我们就直接返回这两个个位数乘法的结果。层层返回,最终得到 N 位数的乘法结果。

时间复杂度

我们平常使用的长乘法,是 O (n ^ 2) 的时间复杂度。比如两个 N 位数相乘,我们需要将每一位按规则相乘,所以需要计算  N * N 次乘法。而使用  Karatsuba 算法每层需要计算三次乘法,两次加法,以及若干次加法,每使用一次 karatsuba 算法,乘法规模就下降一半。

所以,对于两个 n =  2 ^ K 位数乘法运算,我们需要计算 3 ^ k 次乘法运算。而 K = log n(底数为 2), 3 ^ K = 3 ^ log n = 2  ^ (log 3 * log n) = 2 ^ (log n * log 3) = n ^ log 3 (底数为 2)。

代码实现

from math import log2, ceil
 
def pad(string: str, real_len: int, max_len: int) -> str:
  pad_len: int = max_len - real_len
  return f"{'0' * pad_len}{string}"
 
 
def kara(n1: int, n2: int) -> int:
  if n1 < 10 or n2 < 10:
    return n1 * n2
  n1_str: str = str(n1)
  n2_str: str = str(n2)
  n1_len: int = len(n1_str)
  n2_len: int = len(n2_str)
  real_len: int = max(n1_len, n2_len)
  max_len: int = 2 ** ceil(log2(real_len))
  mid_len: int = max_len >> 1
  n1_pad: str = pad(n1_str, n1_len, max_len)
  n2_pad: str = pad(n2_str, n2_len, max_len)
  p: int = int(n1_pad[:mid_len])
  q: int = int(n1_pad[mid_len:])
  r: int = int(n2_pad[:mid_len])
  s: int = int(n2_pad[mid_len:])
  u: int = kara(p, r)
  v: int = kara(q-p, r-s)
  w: int = kara(q, s)
  return u * 10 ** max_len + (u+v+w) * 10 ** mid_len + w

输出结果:

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

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

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python中模块查找的原理与方法详解
Aug 11 Python
利用numpy实现一、二维数组的拼接简单代码示例
Dec 15 Python
Python用 KNN 进行验证码识别的实现方法
Feb 06 Python
对python numpy数组中冒号的使用方法详解
Apr 17 Python
Python datetime包函数简单介绍
Aug 28 Python
解决Django layui {{}}冲突的问题
Aug 29 Python
python flask搭建web应用教程
Nov 19 Python
python+Selenium自动化测试——输入,点击操作
Mar 06 Python
django的403/404/500错误自定义页面的配置方式
May 21 Python
利用Python实现斐波那契数列的方法实例
Jul 26 Python
使用Python爬取Json数据的示例代码
Dec 07 Python
python 使用OpenCV进行简单的人像分割与合成
Feb 02 Python
Python对接 xray 和微信实现自动告警
Sep 17 #Python
Python计算两个矩形重合面积代码实例
Sep 16 #Python
详解Python3 pickle模块用法
Sep 16 #Python
python scipy卷积运算的实现方法
Sep 16 #Python
python 三元运算符使用解析
Sep 16 #Python
python 利用pywifi模块实现连接网络破解wifi密码实时监控网络
Sep 16 #Python
Python循环实现n的全排列功能
Sep 16 #Python
You might like
PHP经典的给图片加水印程序
2006/12/06 PHP
phpMyAdmin 安装教程全攻略
2007/03/19 PHP
JavaScript中的View-Model使用介绍
2011/08/11 Javascript
关于JS控制代码暂停的实现方法分享
2012/10/11 Javascript
window.open的页面如何刷新(父页面)上层页面
2012/12/28 Javascript
setInterval,setTimeout与jquery混用的问题
2013/04/08 Javascript
Event altKey,ctrlKey,shiftKey属性解析
2013/12/18 Javascript
js判断浏览器类型为ie6时不执行
2014/06/15 Javascript
JavaScript开发者必备的10个Sublime Text插件
2016/02/27 Javascript
AngularJS指令用法详解
2016/11/02 Javascript
JS实现二叉查找树的建立以及一些遍历方法实现
2017/04/17 Javascript
angular使用post、get向后台传参的问题实例
2017/05/27 Javascript
Vue resource中的GET与POST请求的实例代码
2017/07/21 Javascript
JS中跳出循环的示例代码
2017/09/14 Javascript
在HTML文档中嵌入JavaScript的四种方法
2018/05/07 Javascript
JavaScript函数apply()和call()用法与异同分析
2018/08/10 Javascript
微信小程序解析富文本过程详解
2019/07/13 Javascript
解决layer.open弹出框不能获取input框的值为空的问题
2019/09/10 Javascript
JS函数进阶之prototy用法实例分析
2020/01/15 Javascript
关于vue 结合原生js 解决echarts resize问题
2020/07/26 Javascript
Python3通过Luhn算法快速验证信用卡卡号的方法
2015/05/14 Python
Django中对数据查询结果进行排序的方法
2015/07/17 Python
Python编程对列表中字典元素进行排序的方法详解
2017/05/26 Python
Anaconda多环境多版本python配置操作方法
2017/09/12 Python
查看python安装路径及pip安装的包列表及路径
2019/04/03 Python
对Pytorch神经网络初始化kaiming分布详解
2019/08/18 Python
python将字典列表导出为Excel文件的方法
2019/09/02 Python
基于Python的接口自动化读写excel文件的方法
2021/01/15 Python
小学少先队活动方案
2014/02/18 职场文书
入党自荐书范文
2014/03/09 职场文书
小学领导班子对照材料
2014/08/23 职场文书
幼儿园园长工作总结2015
2015/05/25 职场文书
疾病证明书
2015/06/19 职场文书
网吧员工管理制度
2015/08/05 职场文书
Python预测分词的实现
2021/06/18 Python
python微信智能AI机器人实现多种支付方式
2022/04/12 Python