算法的重要性,我就不多说了吧,想去大厂,就必须要经过基础知识和业务逻辑面试+算法面试。所以,为了提高大家的算法能力,这个公众号后续每天带大家做一道算法题,题目就从LeetCode上面选 !

今天和大家聊的问题叫做 字符串相乘,我们先来看题面:

https://leetcode-cn.com/problems/multiply-strings/

Given two non-negative integers num1 and num2 represented as strings, return the product of num1 and num2, also represented as a string.

题意

给定两个以字符串形式表示的非负整数 num1 和 num2,返回 num1 和 num2 的乘积,它们的乘积也表示为字符串形式。

样例

示例 1:
输入: num1 = "2", num2 = "3"
输出: "6"
示例 2:
输入: num1 = "123", num2 = "456"
输出: "56088"

说明:

  • num1 和 num2 的长度小于110。

  • num1 和 num2 只包含数字 0-9。

  • num1 和 num2 均不以零开头,除非是数字 0 本身。

  • 不能使用任何标准库的大数类型(比如 BigInteger)或直接将输入转换为整数来处理。

解题

来源:https://www.cnblogs.com/techflow/p/12544184.html

高精度与打竖式

这就需要我们的高精度算法出场了,其实严格说起来高精度并不是一种算法,而是一种思想。这个思想非常朴素,我敢保证我们每一个人都学过。还记得小学的时候,我们计算多位数的乘法是怎么算的吗?大家应该都不陌生才对,就是打竖式,like this:


我们人类要打竖式是因为我们只能计算一位数以内的加减乘除,超过一位的人脑不能直接计算,我们就需要用纸笔记录下来进行计算。纸笔计算的方法很简单,就是一位一位地计算,用每一位数字依次去计算乘法,最后再移位相加起来就得到结果了。比如在上图的第一个例子当中,我们要计算15 * 16,我们先计算6 * 15的结果,再计算1 * 15,最后将两个结果错位相加,就得到了答案。我们要错位的原因也很简单,因为我们在计算15 * 1的时候,其实背后代表的是15 * 10。我们继续拆分问题,当我们计算6和15相乘的时候,又是怎么计算的呢?顺着这个思路,整个过程可以进一步被划分成先计算6和5相乘,再计算6和1相乘。最后,我们把两个较大数字的相乘拆分成了在每一位上的数字相乘。到了这里,剩下的就简单了,也就是说我们可以把这两个很大的数字用两个数组来存储,数组当中的每一位存储数字上的一位。比如我们要计算123 * 224, 我们的第一个数组是[1, 2, 3],我们的第二个数组是[2, 2, 4]。我们仿照乘法竖式中的方法计算这两个数组当中两两的乘积,并将它们拼装成答案。

123   *  224  ____________492246246  ____________27552

同样我们用数组来存储中间和最后的结果,最后的结果就是:[2, 7, 5, 5, 2]。由于题目需要我们要返回的是字符串,所以我们还需要将数组里的内容再拼接成字符串。这种用数组来模拟数字进行加减乘除运算的方法就叫做高精度算法,相信大家也都看到了,严格说起来这并不是一个算法,而只是一种思想。今天的题目出的是乘法,我们利用同样的方法也可以计算加减和除法。其中加减法非常简单,而除法则要复杂得多,也是高精度当中最难实现的部分。这里我们不做过多的拓展,计算的方法同样是打竖式,感兴趣的同学可以自行实现。

进位和前导零

当我们理清楚了打竖式的方法之后,我们还要面临进位和前导零的问题。进位应该很容易理解,我们需要在计算乘法的时候判断当前位置的元素是否大于等于10,如果超过10的话,我们则需要进行进位。我们只需要将它除以10,得到的结果就是我们需要进位的值。除此之外就是前导零的问题,我们都知道除了零以外的合法数字是不允许首位出现0的,但是由于我们计算的是乘法,所以当其中某一个数为0会得到整体的结果为0,但是表示在数组当中则是多个0.举个简单的例子,比如123 * 0,最后得到的应该是0,但是由于我们用数组表示了乘法运算当中的每一位,并且还进行了加法计算,所以会导致出现000的结果。这种情况我们要做特殊的处理,不过这也不复杂。最后我们把上面所有的思路都整理一下,就可以得到结果了。我们来看下代码:

class Solution:def multiply(self, num1: str, num2: str) -> str:# 将字符串转化成数组# 翻转数组,因为我们用第0位表示个位        arr1 = [ord(i) - ord('0') for i in num1][:: -1]        arr2 = [ord(i) - ord('0') for i in num2][:: -1]# 创建结果数组,可以证明结果的长度最多是n + m        n, m = len(arr1), len(arr2)        ret = [0for i in range(n + m + 1)]for i in range(n):for j in range(m):# 按位相乘,计算进位                ret[i + j] += arr1[i] * arr2[j]if ret[i+j] >= 10:                    ret[i+j+1] += ret[i+j] // 10                    ret[i+j] %= 10# 最后把数组再转化成字符串返回# 去除前导零        result = ''.join(map(str, ret))[::-1].lstrip('0')return result if len(result) > 0else'0'

今天的题只是Medium难度,并不算困难,会选这题的原因主要是为了高精度算法。高精度算法本身并不难,也并不常用即使是在算法比赛当中也不常见。但是它给了我们一个思路,当我们要计算的数值超过计算机目前承载能力的时候,我们还有什么方法?当然这题我们也可以取巧,因为Python当中内置了大整数,当它检测到我们的计算结果超过范围的时候,会自动转化成大整数来进行计算。所以这题如果我们使用Python,可以只用几行代码搞定:

class Solution:def multiply(self, num1: str, num2: str) -> str:        num1 = int(num1)        num2 = int(num2)return str(num1 * num2)

好了,今天的文章就到这里,如果觉得有所收获,请顺手点个在看或者转发吧,你们的支持是我最大的动力。

上期推文:

LeetCode1-20题汇总,速度收藏!LeetCode21-40题汇总,速度收藏!LeetCode刷题实战40:组合总和 IILeetCode刷题实战41:缺失的第一个正数LeetCode刷题实战42:接雨水

leetcode数组汇总_LeetCode刷题实战43:字符串相乘相关推荐

  1. leetcode数组汇总_LeetCode刷题实战118:杨辉三角

    算法的重要性,我就不多说了吧,想去大厂,就必须要经过基础知识和业务逻辑面试+算法面试.所以,为了提高大家的算法能力,这个公众号后续每天带大家做一道算法题,题目就从LeetCode上面选 !今天和大家聊 ...

  2. leetcode数组汇总_LeetCode刷题:前言

    LeetCode刷题:前言 前言 作为一个对编程超级不通的小白,在2020年11月开始打算正式的刷LeetCode. (PS:前面有刷过,但是都是随机,看心情乱刷的,刷完后也没有什么感觉,该不会的还是 ...

  3. 数组翻转_LeetCode刷题实战151:翻转字符串里的单词

    算法的重要性,我就不多说了吧,想去大厂,就必须要经过基础知识和业务逻辑面试+算法面试.所以,为了提高大家的算法能力,这个公众号后续每天带大家做一道算法题,题目就从LeetCode上面选 !今天和大家聊 ...

  4. laravel数组转换为字符串_LeetCode刷题实战108:将有序数组转换为二叉搜索树

    算法的重要性,我就不多说了吧,想去大厂,就必须要经过基础知识和业务逻辑面试+算法面试.所以,为了提高大家的算法能力,这个公众号后续每天带大家做一道算法题,题目就从LeetCode上面选 ! 今天和大家 ...

  5. leetcode 高薪_LeetCode刷题实战69:x 的平方根

    算法的重要性,我就不多说了吧,想去大厂,就必须要经过基础知识和业务逻辑面试+算法面试.所以,为了提高大家的算法能力,这个公众号后续每天带大家做一道算法题,题目就从LeetCode上面选 ! 今天和大家 ...

  6. 子集和问题 算法_LeetCode刷题实战90:子集 II

    算法的重要性,我就不多说了吧,想去大厂,就必须要经过基础知识和业务逻辑面试+算法面试.所以,为了提高大家的算法能力,这个公众号后续每天带大家做一道算法题,题目就从LeetCode上面选 ! 今天和大家 ...

  7. left join缺失右括号_LeetCode刷题实战31:最长有效括号

    来源: https://www.cnblogs.com/techflow/p/12393742.html 算法的重要性,我就不多说了吧,想去大厂,就必须要经过基础知识和业务逻辑面试+算法面试.所以,为 ...

  8. list元素求和_LeetCode刷题实战82:删除排序链表中的重复元素 II

    算法的重要性,我就不多说了吧,想去大厂,就必须要经过基础知识和业务逻辑面试+算法面试.所以,为了提高大家的算法能力,这个公众号后续每天带大家做一道算法题,题目就从LeetCode上面选 ! 今天和大家 ...

  9. c++刷题_LeetCode刷题实战70:爬楼梯

    算法的重要性,我就不多说了吧,想去大厂,就必须要经过基础知识和业务逻辑面试+算法面试.所以,为了提高大家的算法能力,这个公众号后续每天带大家做一道算法题,题目就从LeetCode上面选 ! 今天和大家 ...

最新文章

  1. WebForm页面生命周期及asp.net运行机制
  2. 要锻炼二手交换的能力
  3. 搞嵌入式Linux,做底层还是应用?底层要掌握哪些技能?
  4. iOS核心动画之CALayer-隐式动画
  5. cf1491C. Pekora and Trampoline
  6. 净值:测试编码器/解码器
  7. 集群应用服务器环境中会话管理(复制)的Oracle Coherence最佳实践
  8. scss-@for 指令
  9. java处理高并发高负载类网站问题
  10. java 防止sql注入的方法(非原创)
  11. kali rolling source 16.1~~18.2
  12. Vue.js刷新当前页面
  13. c语言文件读入到链表函数fscanf,【求解答】c关于把文件数据放进链表,并将链表遍历...
  14. 数学建模系列-优化模型---(一)规划模型
  15. java毕业设计——基于java+Spring+SSH的CRM客户关系管理系统设计与实现(毕业论文+程序源码)——CRM客户关系管理系统
  16. excel小技巧之如何提取指定字符之前的字段
  17. GIDC全球互联网数据大会:政企应用如何构筑安全合规的互联内容分发加速?
  18. Power BI中计算同比、环比
  19. 红米K40 面具root教程
  20. jdk1.8——Base64

热门文章

  1. 【转】c#数字图像处理(二)彩色图像灰度化,灰度图像二值化
  2. 【转】ABP源码分析二十二:Navigation
  3. 第十四节:Lambda、linq、SQL的相爱相杀(3)
  4. LeetCode每日打卡 - 汉明距离
  5. 【LeetCode - 42. 接雨水】
  6. 【LightOJ - 1038】Race to 1 Again(概率dp,数学期望)
  7. 【LightOJ - 1027】A Dangerous Maze(概率dp,数学期望)
  8. 【HDU - 6183】Color it(CDQ分治 或 动态开点线段树)
  9. 【FZU - 2140 】Forever 0.5 (计算几何,构造)
  10. 【51nod - 1065】 最小正子段和( 前缀和排序 )