当前位置: 首页 > news >正文

大数运算

大数运算

       大数运算的实现方法主要有下面几种:

1)        用字符串表示大数。将大数用十进制字符数组表示,然后依照“竖式计算”的思想进行计算。这样的方法比較easy理解,可是计算效率非常低。

2)        将大数看成二进制流进行处理。使用各种位运算和逻辑操作来实现打算的运算。该方法设计复杂,可读性较差,并且难以调试。

3)        将大数表示成一个n进制数组。n的取值越大,数组的大小越小,这样能够缩短运算的时间及空间复杂度,提高算法的效率。在32位系统中,n能够取2^32,这时每一位的取值范围是0~0xffffffff

 

以下就针对第3)种方法进行描写叙述。在RSA中涉及的大数通常都大于0,所以为了简化问题,如果运算过程中全部大数均都大于0的。

n=2^32时,大数的每一位恰好是unsigned long 的范围,考虑到加法和乘法中进位时的溢出现象不好推断,能够选取long long型(gcc, 若是VC++则为__int64)。并且对于每一个大数,包括一个变量标志其符号,一个变量来表示其长度,一个数组来存储每一位的值。于是,能够用以下结构体表示一个大数:

typedef struct LargeNumber

{

    bool tag;                          // 标志大数的符号 true  正数  false 负数

    int length;                         // 记录大数的长度

    long long bigInt[SIZE];              // 记录大数各位的值

 

    LargeNumber( const bool& t=true,const int& len=0)

    {

        tag = t;

        length = len;

        for( int i=0; i<SIZE; ++i )

        {

            bigInt[i]= 0;

        }

    }

};

 

(一)    大数加法

如果在加法中两个操作数都是大于0的。依照“竖式计算”的思想,首先将两操作数低位对齐,然后从最低位開始按“位”相加,当“位”相加的结果大于2^32-1时做进位处理(carry=1),否则不进位(carry=0)。

符号演示:op1=ABCD op2=EFG

A  B  C  D

+  E  F  G

-------------------

H  I  J  K   L

 

初始化carry=0

当中,若D+G+carry>0xffffffff L=D+G+carry-0xffffffff-1, carry=1

                                          否则L=D+G+carry carry=0

依照上述方法计算K,J,I,H

 

比如:0x1  0x1  0x1  + 0xffffffff 0xffffffff; 初始carry=0

运算过程:

10x1+0xffffffff+0>0xffffffff, 所以计算结果的最低位result.bigInt[0]=0x0,carry=1;

20x1+0xffffffff+1>0xffffffff,所以result.bigInt[1]= 0x1+0xffffffff+1-0xffffffff-1=1,carry=1;

30x1+1<0xffffffff,所以result.bigInt[2]=2;result.length=2;

 

终于结果为2  1  0

 

(二)    大数减法

为了简化计算,如果被减数总是不小于减数,这样计算的终于结果总是大于等于0。基本思路和大数加法基本一致,减法中可能须要借位,定义并初始借位变量borrow=0。显然,borrow要么等于0,要么等于1

 

比如:0x1 0x1 0x1 – 0xffffffff  0xffffffff; 初始borrow=0;

计算过程:

       (1) 0x1<0xfffffff+borrow,这时须要借位

result.bigInt[0]= 0xffffffff-(0xffffffff+0-0x1)+1=2,borrow=1;

       (2)0x1<0xffffffff+borrow,于是

result.bigInt[1]= 0xffffffff-(0xffffffff+1-0x1)+1=1,borrow=1;

       (3)0x1=0+borrow,于是result.bigInt[2]=0, borrow=0;

 

终于结果为:1  2

(三)    大数乘法

如果乘法的两个操作数均为正数。依照“竖式计算”的思想,大数的乘法能够借助大数的加法,用乘数的每一位乘以被乘数,然后将每一次的计算结果相加。在每位相乘的过程中依旧存在进位现象,并且此时进位不仅仅是1,还存在更大的数。

 

过程演示:

A   B

*   C   D

---------------

           E   F   G

  +  H  I   J

----------------------------

 K  M  N   O  P

 

当中 B*D+carry>0xffffffff,

那么进位G=(B*D+carry)%0xffffffffcarry=(B*D+carry)/0xffffffff;

否则,G=B*D+carry,carry=0;

依照上述计算原则计算,F E J I H.并依照大数加法的计算方法计算P O N M K

 

比如: 0x1  0x1  0x1 * 0xffffffff  0xffffffff.

计算过程:

10x1 0x1 0x1 * 0xffffffff =0xffffffff 0xffffffff 0xffffffff

20x1 0x1 0x1 * 0xffffffff =0xffffffff 0xffffffff 0xffffffff, 因为这是乘数的第一位,所以结果“扩展一位”,变成0xffffffff 0xffffffff 0xffffffff 0x0

3)计算0xffffffff 0xffffffff 0xffffffff + 0xffffffff 0xffffffff 0xffffffff 0x0

4)结果为1 0 0xffffffff 0xfffffffe 0xffffffff

 

(四)    大数除法
除法建立在乘法的基础上,循环搜索num 使得num*op1==op2.效率较低。

代码下载链接:http://shamohua.download.csdn.net/

里面还有字符串实现的代码。

相关文章:

  • 艰苦的RAW格式数据恢复之旅
  • 应用程序迁移到云平台的最佳实践
  • PHP 操作mongodb api大部分方法
  • tcpdump -i eth0 -n -vvv src or dst port 443
  • Exchange 2013 ServerComponent状态异常处理
  • Oracle Minus 取差集
  • winform窗体取消最大化双击标题最大化
  • 初识 Cloudera Impala
  • php文件遍历
  • 如何让开机时,电脑用户自动登录?
  • 控制台登录,提示证书错误
  • 不高兴的小明
  • iis浏览网页时提示无法显示 XML 页
  • mvc和三层架构到底有什么区别
  • Java代码Bug分析插件 FindBugs
  • Android Volley源码解析
  • Brief introduction of how to 'Call, Apply and Bind'
  • docker-consul
  • linux学习笔记
  • orm2 中文文档 3.1 模型属性
  • ReactNative开发常用的三方模块
  • webpack项目中使用grunt监听文件变动自动打包编译
  • Zepto.js源码学习之二
  • 回顾 Swift 多平台移植进度 #2
  • 开发了一款写作软件(OSX,Windows),附带Electron开发指南
  • 从如何停掉 Promise 链说起
  • #{}和${}的区别?
  • #HarmonyOS:基础语法
  • (26)4.7 字符函数和字符串函数
  • (待修改)PyG安装步骤
  • (二)基于wpr_simulation 的Ros机器人运动控制,gazebo仿真
  • (附源码)spring boot校园拼车微信小程序 毕业设计 091617
  • (力扣记录)1448. 统计二叉树中好节点的数目
  • (六) ES6 新特性 —— 迭代器(iterator)
  • (七)c52学习之旅-中断
  • ***php进行支付宝开发中return_url和notify_url的区别分析
  • .bashrc在哪里,alias妙用
  • .NET Core IdentityServer4实战-开篇介绍与规划
  • .NET Framework Client Profile - a Subset of the .NET Framework Redistribution
  • .NET 表达式计算:Expression Evaluator
  • .NET 事件模型教程(二)
  • .NET/C# 的字符串暂存池
  • .NET委托:一个关于C#的睡前故事
  • .Net下的签名与混淆
  • .NET中 MVC 工厂模式浅析
  • .secret勒索病毒数据恢复|金蝶、用友、管家婆、OA、速达、ERP等软件数据库恢复
  • //解决validator验证插件多个name相同只验证第一的问题
  • ?.的用法
  • @FeignClient 调用另一个服务的test环境,实际上却调用了另一个环境testone的接口,这其中牵扯到k8s容器外容器内的问题,注册到eureka上的是容器外的旧版本...
  • @ModelAttribute使用详解
  • @TableLogic注解说明,以及对增删改查的影响
  • [ 云计算 | Azure 实践 ] 在 Azure 门户中创建 VM 虚拟机并进行验证
  • []使用 Tortoise SVN 创建 Externals 外部引用目录
  • [20161214]如何确定dbid.txt
  • [bzoj 3534][Sdoi2014] 重建