大数乘法问题(C++版)

近日参加一个笔试,遇到大数乘法问题,这是一个经典的算法题。所谓大数乘法问题其实就是这样的:输入两个整数,要求输出这两个数的乘积。输入的数字可能超过计算机内任何数据的存储范围。这里主要需要注意的点就是需要使用字符串或者字符数组来存储这两个大数以及他们的结果,还有乘法计算过程中存在乘法进位和加法进位。

我先自己尝试写了一个答案,思路就是模拟手写竖式乘法。

    static string MYMULTIPLY(const string &number1, const string &number2)
    {
        int length1 = number1.size();
        int length2 = number2.size();
        string result = "";

        if (length1 == 0 || length2 == 0)
        {
            result = "error";
            return result;
        }

        int *iresult;
        iresult = (int*)malloc(sizeof(int) * (length1 + length2 + 1));
        memset(iresult, 0, sizeof(int) * (length1 + length2 + 1));

        for(int i = length1 - 1, x = 0; i >= 0; i--, x++)
        {
            int numA = number1[i] - 48;
            int value1 = 0;
            int value2 = 0;
            int multiFlag = 0;//乘法进位数
            int addFlag = 0;//加法进位数

            for(int j = length2 - 1, y = 0; j >= 0; j--, y++)
            {
                int numB = number2[j] - 48;
                value1 = numA * numB + multiFlag;
                multiFlag = value1 / 10;
                value1 = value1 % 10;
                value2 = (iresult[x+y]) + value1 + addFlag;
                addFlag = value2 / 10;
                iresult[x+y] = (value2 % 10); 
            }
            iresult[x + length2] += (multiFlag + addFlag);
        }

        //逆序
        int i = 0;
        for(i = length1 + length2 - 1; i >= 0; i--)
        {
            if(iresult[i] != 0)
                break;
        }

        for(; i >= 0; i--)
        {
            result = result + (char)(iresult[i]+48);
        }

        free(iresult);

        return result;
    }

这个方法虽然能得出结果,但是太难看了,特别是计算每一位相乘的结果和进位的那一段代码。之后,我就把这段代码改进了一下,将每一位之间的乘法和进位计算分开来。

    static string MULTIPLY(string number1, string number2)
    {
        int i, j;
        int *iresult;
        int length1 = number1.size();
        int length2 = number2.size();
        string result = "";

        reverse(number1.begin(), number1.end());
        reverse(number2.begin(), number2.end());

        if (length1 == 0 || length2 == 0)
        {
            result = "error";
            return result;
        }

        iresult = (int*)malloc(sizeof(int) * (length1 + length2 + 1));
        memset(iresult, 0, sizeof(int) * (length1 + length2 + 1));

        //每一位相乘
        for(i = 0; i < length1; i++)
        {
            for(j = 0; j < length2; j++)
            {
                iresult[i+j] += ((number1[i] - 48) * (number2[j] - 48));
            }
        }

        //进位运算
        int carry = 0;
        for(i = 0; i < length1 + length2; i++)
        {
            int value = iresult[i] + carry;
            iresult[i] = value % 10;
            carry = value / 10;
        }

        for(i = length1 + length2 - 1; i >= 0; i--)
        {
            if(iresult[i] != 0)
            break;
        }

        for(; i >= 0; i--)
        {
            result = result + (char)(iresult[i]+48);
        }

        free(iresult);

        if(result == "") result = "0";
        return result;
    }

当然这个问题还有分治乘法,快速傅里叶等算法,这个就不是在笔试或者面试的时候能快速写出来的了。有兴趣大家可以继续深入了解下。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 一、计算机的发展史 01改变世界:没有计算器的日子怎么过——手动时期的计算工具 所谓计算机,顾名思义,就是用于计...
    文思汇集阅读 2,751评论 1 8
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,742评论 18 399
  • 贪心算法 贪心算法总是作出在当前看来最好的选择。也就是说贪心算法并不从整体最优考虑,它所作出的选择只是在某种意义上...
    fredal阅读 9,266评论 3 52
  • 2017年4月26日 天气雨 星期三 经典有好几本,有易经、有唐诗三百首、有老子、还有幼雪琼林,我们还...
    琦琦花仙子小月阅读 164评论 1 3
  • 如果时钟的存在是为了提醒萎靡不振的人们不断赛跑。那过往是否可以容纳未来的光。 如果生活的瞬息万变是在告知我们需要勇...
    蚕食阅读 369评论 3 1