[转]高精度乘法计算

转载自: 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都可以计算他们的乘积

技术追求卓越 梦想创造未来

转载于:https://www.cnblogs.com/lastavengers/p/3871942.html

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值