[转]高精度乘法计算
转载自: Daywei 高精度乘法计算
高精度乘法计算基础
1.高精度浮点运算方法
高精度浮点(Floating Point,FP)运算可以转换成整数型运算。由于高精度浮点数可以看成是由整数部分(Integer Part,IP)与小数部分(Decimal Part,DP)的组合,因此其乘法可以看成以下3种运算的组合,即整数x整数(IxI)、整数x小数(IxD)和小数x小数(DxD)。用表达式表示, 则FP1*FP2=IP1*IP2+(IP1*DP2+IP2*DP1)+DP1*DP2
(1)对于IxI型运算,所得的结果仍是整数
(2)对于DxD型运算,所得结果仍是小数
(3)对于IxD型运算,所得到的计算结果是一个浮点数,即可能包括整数和小数。
此时可以使用小数部分的每个单元分别乘以整数部分。这样对于n位小数,可以得到拥有n个数字的字符串。把第d位乘以整数部分所得到的所有数字向低位移动d位,则可以得到整数和小数部分的数字位,然后进行求和,即可以得到所需要的计算结果。
2.高精度乘法与移位计算
由于在高精度计算中采用了数组或链表作为数值储存单元,因此可以使用移位计算代替乘法或除法运算。在对二进制计算中,对于一个数值,每个单元向 高位移动一位,得到的数值结果等于这个数乘以2,而向低位移动一位时,其结果等于这个数值除以2,如8>>1=4,8<< 1=16.同理,在高精度计算时,若进制基数为S,而向低位移动一位时,其结果等于这个数值除以S。注意,在移位后应把空出的低位设置为0.使用移位操作 代替乘法或除法计算,可以大大地提高计算效率。
高精度整数乘法
1.高精度与单精度整数乘法
这种乘法运算与高精度加法类似,所不同的是,对于高精度数字由低位到高位逐位乘以单经读书。设高精度的数位数字为a[i],单精度数为b,则可以把a[i]*b除以进制的玉树作为更新a[i]的值,把其商数作为进位。
void mults( int a[], int na, int b, int c[], int * nc) { int i,s,carry=0; for (i=0;i<na;i++) //高精度与单精度乘法 { s=carry+a[i]*b; //本位数乘以单精度整数,在加上进位 c[i]=s%10; //计算本位数字 carry=s/10; //计算进位 } while (carry!=0) //对进位进行处理 { c[na]=carry%10; //计算本位 carry=carry/10; //计算进位 na++; //计算结果的数字长度 } *nc=na; //返回结果数字长度 } |
2.高精度与高精度整数乘法
因为当两个高精度整数相乘时,所得到的积的位数最多不会超过最长整数的两倍,所以保存计算结果的变量空间可以设置为最长整数位数的两倍。当i和 j初始值从0开始,乘数的第i位a[i]与被乘数的第j位b[j]相乘时,则其结果可以加入到c[i+j]中,然后再进行进位处理
void multm( int a[], int na, int b[], int nb, int c[], int *nc) { int i,j,s,carry=0; *nc=na+nb; //乘积长度 for (i=0;i<*nc;i++) c[i]=0; //乘积变量初始化 for (i=0;i<na;i++) { for (j=0;j<nb;j++) { c[i+j]=c[i+j]+a[i]*b[j]; //两个整数数位与数位之间相乘 c[i+j+1]=c[i+j+1]+c[i+j]/10; //进位 c[i+j]=c[i+j]%10; //计算本位数字 } } } |
高精度浮点数乘法
1.整数与小数相乘
在不考虑小数点的情况下,先计算整数与每个小数单元中的数字,每次可以 得到一个长整数,在根据当前小数单元的位置,确定对这个长整型的缩小位数。第1位小数,缩小10倍(这里以十进制为例),相当于将此长整数整体向右移动1 位。同理,对于第n为小数,相当于将此长整数整体向右移动n位。由此,每次计算都可以得到一个确定的整数和小数部分。对这些数值求和,即可以得到整数与小 数部分的乘积。
void ixd( int xi[], int ni, int yd[],intnd, int zi[], int zd[]) { int i,n,carry; int x1[MAXLEN],x2[MAXLEN],tmp[MAXLEN]; for (i=0;i<ni;i++) zi[i]=0; for (i=0;i<nd;i++) zd[i]=0; for (i=0;i<nd;i++) //循环小数数字单元 { mults(xi,ni,yd[nd-i-1],tmp,&n); //整数与每个小数位上的数字相乘 rart(tmp,n,i+1,x1,ni,x2,nd); //从新整数中分离出整数和小数 carry=add_int(zd,x2,zd,nd); //小数累加 zi[0]+=carry; //向整数进位 add_int(zi,x1,zi,ni); //整数累加 } } //通过移位把一个整数分解成两个整数,分别表述整数和小数部分。这种移位计算相当于乘法运算 void rpart( int a[], int maxn, int num, int ai[], int ni, int ad[], int nd[]) { int i; for (i=0;i<ni;i++) ai[i]=0; for (i=0;i<nd;i++) ad[i]=0; if (num<=maxn) //缩小位数小于给定的整数长度 { for (i=0;i<maxn-num;i++) ai[i]=a[num+i]; //计算得到整数数字单元中的数字 for (i=0;i<num;i++) ad[nd-num+i]=a[i]; //计算得到小数数字单元中的数字 } else //此情况下,整数部分全为0 { for (i=0;i<maxn;i++) ad[nd-num+i] =a[i]; //计算得到小数数字单元中的数字 } } |
2.两整数相乘与两小数相乘
这两种乘法都可以使用长整数乘法来运算。整数与整数相乘的运算较简单,只需使用前面介绍的整数乘法函数运算即可。对于小数与小数相乘的运算,可以 先不考虑小数点问题,把他们作为两长整数进行运算。如1.85 x 2.123,在计算时,先不考虑小数点,直接计算85 x123=10455,它表示小数乘法0.85x0.123=0.10455 两个小数相乘后仍然是小数,其位数为这两个小数位数相加,因此其长度可能超过指定的小数位长度,此时需要将低位社区,可以使用一个截断字符串函数来进 行。这种截断也便于两小数相加时小数位对齐。在进行两个小数相加或相减时,必须需要注意“小数位对齐”,否则计算将出错
//用于对长度为maxn位的数字,以左为界,取出num位数字 void ltruncate( int a[], int maxn, int b[], int num) { int i; for (i=manx-1;i>=maxn-num;i++) //以左为界,取出num位数字 b[i-maxn+num] =a[i]; } |
3.浮点数与浮点数相乘
void multiply(BIGFLOAT a,BIGFLOAT b,BIGFLOAT *c) { int yi[MAXLEN],yd[MAXLEN]; int xi1[MAXLEN],xd1[MAXLEN]; int xi2[MAXLEN],xd2[MAXLEN]; int nyi,nyd,carry1,carry2; if (a.sign==b.sign) //确定乘积符号 c->sign=0; //乘积为正 else c-.sign=1; //乘积为负 c->ni=a.ni; //确定乘积整数长度 c->nd=a.nd; //确定乘积小数长度 multm(a.a,a.ni,b.a,b.ni,yi,&nyi); //整数x整数 multm(a.d,a.nd,b.d,b.nd,yd,&nyd); //小数x小数 ltruncate(yd,nyd,c->d,c->nd); //小数位对齐 ixd(a.a,a.ni,b.d,b.nd,xi1,xd1); //整数x小数 ixd(b.a,b.ni,a.d,a.nd,xi2,xd2); //小数x整数 carry1=add_int(c->d,xd1,c->d,c->nd); //关于小数部分的整数加法计算 carry2=add_int(c->d,xd2,c->d,c->nd); //关于小数部分的整数加法计算 yi[0]+=carry1+carry2; //小数部分向整数部分进位 add_int(yi,xi1,c->a,c->ni); //整数加法计算 add_int(c->a,xi2,c->a,c->ni); //整数加法计算 } |
因此当任意给定两个高精度的浮点数后,使用以上的multily都可以计算他们的乘积