【算法】算法(模拟、指针等)解决字符串类题目(C++)

文章目录

  • 1. 前言
  • 2. 解决 字符串类算法题
    • 14.最长公共前缀
    • 5.最长回文子串
    • 67.二进制求和
    • 43.字符串相乘

1. 前言

字符串题目有很多种,这里筛选几个考察模拟、双指针等的题目,并用相关算法解决。

2. 解决 字符串类算法题

14.最长公共前缀

在这里插入图片描述

思路

  • 题意分析:题目要求找到字符串数组中的最长公共前缀。
  • 解法一两两比较
    • 遍历数组,每次比较后更新最长公共前缀,并循环比较找最长公共前缀
  • 解法二统一比较
    • 遍历第一个字符串的所有字符,将当前字符与其他字符串相同位置的字符进行比较。
    • 如果发现不匹配的字符或某个字符串已经达到最终长度(即没有更多字符可比较),则返回第一个字符串的前缀子串

代码

  • 解法一:
class Solution {
public:string longestCommonPrefix(vector<string>& strs) {string ret = strs[0];// 解法一:两两比较,找公共前缀for(int i = 0; i < strs.size(); i++){ret = findCommonPrefix(ret, strs[i]);}return ret;}string findCommonPrefix(string &s1, string& s2) {// 找公共前缀string tmp = "";for(int i = 0; i < min(s1.size(), s2.size()); ++i){if(s1[i] == s2[i])tmp += s1[i];elsebreak;}return tmp;}
};
  • 解法二:
class Solution {
public:string longestCommonPrefix(vector<string>& strs) {// 解法二:统一比较for(int i = 0; i < strs[0].size(); ++i) // 遍历第一个字符串的所有字符{char tmp = strs[0][i]; // tmp与其他字符比较for(int j = 0; j < strs.size(); ++j){if(tmp != strs[j][i] || i == strs[j].size()) // 如果字符不匹配or有字符串最终长度截止到当前位置   return strs[0].substr(0, i);}}return strs[0];}
};

5.最长回文子串

在这里插入图片描述

思路

在这里插入图片描述

  • 解法中心扩展算法
    • 如图所示,我们遍历数组,依次固定数组每一位,通过左右两指针找最长回文串。
  • 细节注意
    • 奇数长的回文串与偶数长的回文串计算时,两指针起始位置不同,所以我们分别进行计算。 在这里插入图片描述

代码

string longestPalindrome(string s) {// 遍历每一位,双指针按从左右方向移动,并比较int len = 0, n = s.size(), begin = 0; // begin存储最长子串的开始位置for(int i = 0; i < n; ++i){// 进行奇数长回文串的操作判定int left = i, right = i;while((left >= 0 && right <= n) && s[left] == s[right]) // 左右指针每次各移一步left--, right++;if(right - left - 1 > len) // 如果此时回文串长度>len,更新结果{len = right - left - 1;begin = left + 1;}// 进行偶数长回文串的判定left = i, right = i + 1;while(left >= 0 && right <= n && s[left] == s[right]) // 左右指针每次各移一步left--, right++;if(right - left - 1 > len){len = right - left - 1;begin = left + 1;}}return s.substr(begin, len);
}

67.二进制求和

在这里插入图片描述

思路

在这里插入图片描述

  • 解法模拟二进制列式相加的过程
    1. 分别用两指针遍历两字符串,每次用变量carry累加二进制每一位
    2. 后将此次carry加到最终结果中,carry /= 2
    3. 由于字符串ret是逐渐累加结果的,翻转后的字符串才是二进制顺序

代码

string addBinary(string a, string b) {int carry = 0; // 记录是否有进位int cur1 = a.size()-1, cur2 = b.size()-1;string ret = "";while(cur1 >= 0 || cur2 >= 0 || carry){if(cur1 >= 0) carry += a[cur1--] - '0';if(cur2 >= 0) carry += b[cur2--] - '0';ret += carry % 2 + '0'; // carry%2即为相加的和carry /= 2; // 下一位的进位}// 由于字符串是逐渐累加结果的,翻转后的字符串才是二进制顺序reverse(ret.begin(), ret.end());return ret;
}

43.字符串相乘

在这里插入图片描述

思路

  • 题意分析:要求求出 两个字符串表示的整数 的乘积,且不得使用库函数直接进行整形和字符串的转换。
  • 解法模拟列式相乘的过程
    1. 与上题类似,我们对两字符串首先进行不进位相乘
      • 将输入的两个字符串逆序,从个位开始计算
      • 对应位置上的数字相乘,并将结果存储在临时数组中
      • 后将所有相乘结果相加
    2. 处理进位
      • 定义一个变量carry来记录进位数,然后从数组的第一位开始,将当前位置上的数字与carry相加,得到当前位置上的数字的和,并更新carry为下一位的进位数
      • 将每一位上的结果转换为字符,并添加到结果字符串ret中
      • 去掉结果字符串ret的前导零,并将其逆序,得到最终的结果
        在这里插入图片描述

代码

string multiply(string num1, string num2) {// 解法:模拟列式运算过程// 1. 逆序字符串,从个位开始计算reverse(num1.begin(), num1.end());reverse(num2.begin(), num2.end());// 2. 不进位相乘后相加int m = num1.size(), n = num2.size();vector<int> tmp(m + n - 1);for(int i = 0; i < m; ++i)for(int j = 0; j < n; ++j)tmp[i + j] += (num1[i] - '0') * (num2[j] - '0');// 3. 处理进位string ret = "";int cur = 0, carry = 0;while(cur < m + n - 1 || carry){if(cur < m + n - 1) carry += tmp[cur++]; // 记录当前位置元素ret += (carry % 10) + '0'; // ret加上个位carry /= 10; // 下一位的进位数}cout << ret;// 4. 去掉前导0while(ret.size() > 1 && ret.back() == '0')ret.pop_back();reverse(ret.begin(), ret.end());return ret;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/410876.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

统计学-R语言-5.3

文章目录 前言分位数统计量的标准误总结 前言 本篇文章即为概率与分布的最后一篇文章。 分位数 分位数函数是累积分布函数的反函数。 p-分位数是具有这样性质的一个值&#xff1a;小于或等于它的概率为p。 根据定义&#xff0c;中位数即50%分位数。 分位数通常用于置信区间的…

Python4Delphi安装及编译

1.下载或直接克隆python4delphi组件资源到指定目录,我这里下载到Components文件夹下,并对下载的文件夹进行了重命名为(P4D),重命名不是必须的 下载地址:https://github.com/pyscripter/python4delphi 2.安装 2.1在已下载的目录下进入Install文件夹,双击MultiInstaller.exe…

debian12部署Gitea服务之二——部署git-lfs

Debian安装gitlfs: 先更新下软件包版本 sudo apt update 安装 sudo apt install git-lfs 验证是否安装成功 git lfs version cd到Gitea仓库目录下 cd /mnt/HuHDD/Git/Gitea/Repo/hu/testrepo.git 执行lfs的初始化命令 git lfs install客户机Windows端在官网下载并安装Git-Lfs 再…

渗透测试之如何部署和使用Supershell

环境: Supershell v2.0.0 Centos 7.6 docker v. 21 问题描述: 如何部署和使用Supershell 解决方案: 1、下载最新release源码,解压后进入项目目录 wget https://github.com/tdragon6/Supershell/releases/latest/download/Supershell.tar.gz如果在线下很慢,用浏览…

【银行测试】银行项目,信贷/贷款业务测试+常问面试(二)

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 银行测试-信贷&am…

SpringBoot 更新业务场景下,如何区分null是清空属性值 还是null为vo属性默认值?

先看歧义现象 值为null 未传递此属性 所以此时如何区分null 时传递进来的的null&#xff0c;还是属性的默认值null? 引入方案 引入过滤器&#xff0c;中间截获requestBodyData并保存到HttpServletRequest&#xff0c;业务层从HttpServletRequest 获取到requestBodyData辅…

基于springboot+vue的在线拍卖系统(前后端分离)

博主主页&#xff1a;猫头鹰源码 博主简介&#xff1a;Java领域优质创作者、CSDN博客专家、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战 主要内容&#xff1a;毕业设计(Javaweb项目|小程序等)、简历模板、学习资料、面试题库、技术咨询 文末联系获取 项目背景…

易懂的方式讲解ARM中断原理以及中断嵌套方法

ARM有七种模式&#xff0c;我们这里只讨论SVC、IRQ和FIQ模式。 我们可以假设ARM核心有两根中断引脚&#xff08;实际上是看不见的&#xff09;&#xff0c;一根叫 irq pin, 一根叫fiq pin。在ARM的cpsr中&#xff0c;有一个I位和一个F位&#xff0c;分别用来禁止IRQ和FIQ。 先…

vite 打包优化

✨专栏介绍 在当今数字化时代&#xff0c;Web应用程序已经成为了人们生活和工作中不可或缺的一部分。而要构建出令人印象深刻且功能强大的Web应用程序&#xff0c;就需要掌握一系列前端技术。前端技术涵盖了HTML、CSS和JavaScript等核心技术&#xff0c;以及各种框架、库和工具…

Java21 + SpringBoot3集成Spring Data JPA

Java21 SpringBoot3集成Spring Data JPA 文章目录 Java21 SpringBoot3集成Spring Data JPA前言相关技术简介ORM&#xff08;Object-Relational Mapping&#xff0c;对象-关系映射&#xff09;JPA&#xff08;Java Persistence API&#xff0c;Java持久层API&#xff09;Hiber…

Go 语言中高效切片拼接和 GO 1.22 提供的新方法

Table Contents 切片拼接的必要性基本拼接方法及其局限性使用 append 函数高效拼接的策略控制容量和避免副作用利用 Go 1.22 的新特性切片动态扩容的深入理解内存重新分配与数据迁移性能优化策略结论在 Go 语言中,切片拼接是一项常见的操作,但如果处理不当,可能会导致性能问…

Golang通过Gorm操作Mysql时遇到的datetime时区问题

情景描述 golang使用Gorm操作MySQL&#xff0c;MySQL中数据类型是datetime&#xff0c;Golang中用的是time.now。 但是会导致存储的时间与北京时间有8h误差&#xff0c; 显然是没有初始化时区导致。 问题修复 初始化设置时区 参考我自己之前写过的一篇总结——Mysql中多种日…