【算法】活用双指针完成--字符串相乘(双指针,图例详解!!!)

2024-03-31 05:28

本文主要是介绍【算法】活用双指针完成--字符串相乘(双指针,图例详解!!!),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

一、前言

 二、字符串相乘

✨方法一:普通竖式法

✨方法二:优化竖式法

三、共勉 


一、前言

      最近春招已经开始,看周围的同学都在投递一些大厂的实习,某为的手撕代码 --- 字符串相乘某讯的手撕代码 --- 字符串相减等。

     于是专门去 Leetcode 上搜索了一下,发现这类题目是面试常考的题目。只要我们熟练的  掌握四则运算,熟练的掌握字符串的处理,就可以迎刃而解啦!

      常考算法面试题分为:字符串(大数)相加、字符串(大数)相乘、字符串(大数)相减。本次博客主要通过图例和双指针来讲解 ---- 字符串(大数)相乘
      如果想看 字符串相乘可以看这篇文章:字符串相加

 二、字符串相乘

题目链接: 43. 字符串相乘 - 力扣(LeetCode)

 题目分析:

✨方法一:普通竖式法

     该方法基于我们平常的乘法运算,我们一般在草稿纸上列竖式,用乘数的每一位分别乘以被乘数,然后将这些结果加起来得到最终结果,只不过要注意的是:用乘数的各个位与被乘数相乘所得到的这些数在相加时,需要错位相加。

     实际上我们可以将这些错位看成是在数字的后面添上了数字0,这样一来就有规律可循了。在乘数的最后一位(个位)与被乘数相乘的结果后面添上零个0,在乘数的倒数第二位(十位)与被乘数相乘的结果后面添上一个0,在乘数的倒数第三位(百位)与被乘数相乘的结果后面添上两个0,最后将这些结果相加就得到了最终结果。

结论如下: 将乘数的每一位分别与被乘数相乘,并且在乘数的倒数第n位与被乘数相乘的结果后添上n-1个0,再将这些结果相加,即可得到最终结果。

class Solution {
public://字符串相乘string multiply(string num1, string num2) {if (num1 == "0" || num2 == "0")return "0";string retArr; //相乘后的字符串int flag = 0; //标识需要添加的0的个数for (int i = num2.size() - 1; i >= 0; i--) //依次取乘数的倒数第一位、倒数第二位、...{int count = num2[i] - '0'; //乘数当中待与被乘数相乘的那一位的数字string tmp; //存储乘数该位与被乘数相乘后的结果(此时为空)//数count与被乘数相乘,相当于将被乘数加count次到tmp当中for (int j = 0; j < count; j++){tmp = addStrings(tmp, num1);}//在字符串tmp后添上flag个0for (int k = 0; k < flag; k++)tmp.push_back('0');retArr = addStrings(retArr, tmp); //将字符串tmp加到结果字符串retArr当中flag++; //下一次tmp字符串后面需要添加的0的个数增加}return retArr; //返回相乘后的字符串}//字符串相加string addStrings(string s1, string s2) {int next = 0; //标识进位int end1 = s1.size() - 1, end2 = s2.size() - 1;string retArr; //相加后的字符串while (end1 >= 0 || end2 >= 0) //只要有一个字符串未遍历完毕就继续{int num1 = 0; //第一个字符串待相加的数字if (end1 >= 0){num1 = s1[end1] - '0';end1--;}int num2 = 0; //第二个字符串待相加的数字if (end2 >= 0){num2 = s2[end2] - '0';end2--;}int sum = num1 + num2 + next; //两个待相加数字相加后的结果(注意加上进位next)if (sum > 9) //需要进位{next = 1; //标识进位sum -= 10; //进位后该位置的数字}else //无需进位next = 0;retArr.push_back(sum + '0'); //尾插到结果字符串retArr当中}if (next == 1) //还需进位retArr.push_back('1'); //将字符1尾插到结果字符串retArr当中reverse(retArr.begin(), retArr.end()); //逆置字符串retArrreturn retArr; //返回相加后的字符串}
};

✨方法二:优化竖式法

        方法一的做法是将乘数的每一位分别与被乘数相乘(实际上是将被乘数加若干次到空字符串当中),并且在乘数的倒数第n位与被乘数相乘的结果后添上n-1个0,再将这些结果字符串相加,从而得到最终结果。但是整个过程中涉及到较多字符串相加的操作,如果我们使用数组代替字符串存储结果,则可以减少对字符串的操作。

首先,我们取乘数的每一位分别与被乘数的每一位相乘,相乘的结果放在对应位置,但是不进行进位操作,后面统一进行。

到这里先解决一个问题:数组arr应该开辟多大空间才足矣容纳这两个数的乘积?

我们假设被乘数num1的长度为m,乘数num2的长度为n,并且它们均不为0,那么它们乘积的长度为m+n-1或m+n。
证明如下:

  • 如果num1和num2都取最小值,即 num1 = 10m-1,num2 = 10n-1,那么num1和num2的乘积就为10m+n-2,即乘积的长度为m+n-1。
  • 如果num1和num2都取最大值,即num1 = 10m-1,num2 = 10n-1,那么num1和num2的乘积就为10m+n-10m-10n+1,该结果是小于10m+n而大于10m+n-1的,即乘积的长度为m+n。

综上所述:长度分别为m和n的数相乘后的乘积长度为m+n-1或m+n。

      开辟可容纳6个元素的数组arr,然后将对应位置相同的结果相加,相加后的结果放在数组arr的对应位置。

      这里不需要在任何结果后面添0,那我们是如何做到将对应位置相加的结果放在数组arr的对应位置的呢?

    稍作观察,我们可以发现其规律所在,我们只需要将乘数的第 i 位与被乘数的第 j 位相乘的结果加到数组arr中下标为 i+j+1 的位置即可。例如,乘数的5(第1位与)被乘数的3(第2位)相乘的结果应该加到数组arr中下标为4(1+2+1)的位置即可。

    然后我们对数组arr当中的数据从后往前进行进位操作,最终数组当中的数据就是这两个数的乘积。 

      我们将数组当中的数一个个尾插到空字符串当中,该字符串当中存储的便是这两个数的乘积。但是我们应该从数组的第几个位置开始进行尾插呢?

    我们默认从数组中下标为1的位置开始尾插,但如果通过判断发现数组arr当中下标为0的位置的数不为0,那么我们应该从数组当中下标为0的位置开始进行尾插。 

class Solution {
public:string multiply(string num1, string num2) {if (num1 == "0" || num2 == "0")return "0";int m = num1.size(), n = num2.size();vector<int> arr(m + n, 0); //开辟数组arr的大小为m+n,并且全部初始化为0for (int i = n - 1; i >= 0; i--) //取乘数的每一位{int a = num2[i] - '0';for (int j = m - 1; j >= 0; j--) //取被乘数的每一位{int b = num1[j] - '0';arr[i + j + 1] += a*b; //乘数的第i位与被乘数的第j位相乘后的结果放在数组arr中下标为i+j+1的位置}}//从后往前对数组arr进行进位操作int end = m + n - 1;while (end > 0){arr[end - 1] += arr[end] / 10; //进位arr[end] %= 10; //当前位进位后的数end--; //从后往前进行迭代}int flag = 1; //默认有效值从数组arr当中下标为1的位置开始if (arr[0] != 0)flag = 0; //若数组arr当中下标为0的位置的数据不为0,则从第0位开始为有效数据string retArr; //相乘后的字符串//依次将数据尾插到字符串retArr当中for (int i = flag; i < m + n; i++){retArr.push_back(arr[i] + '0');}return retArr; //返回相乘后的字符串}
};

三、共勉 

      以下就是我对 字符串相乘 的理解,如果有不懂和发现问题的小伙伴,请在评论区说出来哦,同时我还会继续更新对 字符串相减 的理解,请持续关注我哦!!!  

这篇关于【算法】活用双指针完成--字符串相乘(双指针,图例详解!!!)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/863504

相关文章

Java中StopWatch的使用示例详解

《Java中StopWatch的使用示例详解》stopWatch是org.springframework.util包下的一个工具类,使用它可直观的输出代码执行耗时,以及执行时间百分比,这篇文章主要介绍... 目录stopWatch 是org.springframework.util 包下的一个工具类,使用它

Java进行文件格式校验的方案详解

《Java进行文件格式校验的方案详解》这篇文章主要为大家详细介绍了Java中进行文件格式校验的相关方案,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、背景异常现象原因排查用户的无心之过二、解决方案Magandroidic Number判断主流检测库对比Tika的使用区分zip

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

springboot security快速使用示例详解

《springbootsecurity快速使用示例详解》:本文主要介绍springbootsecurity快速使用示例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝... 目录创www.chinasem.cn建spring boot项目生成脚手架配置依赖接口示例代码项目结构启用s

Python中随机休眠技术原理与应用详解

《Python中随机休眠技术原理与应用详解》在编程中,让程序暂停执行特定时间是常见需求,当需要引入不确定性时,随机休眠就成为关键技巧,下面我们就来看看Python中随机休眠技术的具体实现与应用吧... 目录引言一、实现原理与基础方法1.1 核心函数解析1.2 基础实现模板1.3 整数版实现二、典型应用场景2

一文详解SpringBoot响应压缩功能的配置与优化

《一文详解SpringBoot响应压缩功能的配置与优化》SpringBoot的响应压缩功能基于智能协商机制,需同时满足很多条件,本文主要为大家详细介绍了SpringBoot响应压缩功能的配置与优化,需... 目录一、核心工作机制1.1 自动协商触发条件1.2 压缩处理流程二、配置方案详解2.1 基础YAML

Python实现无痛修改第三方库源码的方法详解

《Python实现无痛修改第三方库源码的方法详解》很多时候,我们下载的第三方库是不会有需求不满足的情况,但也有极少的情况,第三方库没有兼顾到需求,本文将介绍几个修改源码的操作,大家可以根据需求进行选择... 目录需求不符合模拟示例 1. 修改源文件2. 继承修改3. 猴子补丁4. 追踪局部变量需求不符合很

java中反射(Reflection)机制举例详解

《java中反射(Reflection)机制举例详解》Java中的反射机制是指Java程序在运行期间可以获取到一个对象的全部信息,:本文主要介绍java中反射(Reflection)机制的相关资料... 目录一、什么是反射?二、反射的用途三、获取Class对象四、Class类型的对象使用场景1五、Class

golang 日志log与logrus示例详解

《golang日志log与logrus示例详解》log是Go语言标准库中一个简单的日志库,本文给大家介绍golang日志log与logrus示例详解,感兴趣的朋友一起看看吧... 目录一、Go 标准库 log 详解1. 功能特点2. 常用函数3. 示例代码4. 优势和局限二、第三方库 logrus 详解1.

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.