【Leetcode】string类刷题

Alt

🔥个人主页Quitecoder

🔥专栏Leetcode刷题

Alt

目录

  • 1.仅反转字母
  • 2.字符串中第一个唯一字符
  • 3.验证回文串
  • 4.字符串相加
  • 5.反转字符串I I
  • 6.反转字符串中的单词III
  • 7.字符串相乘
  • 8.把字符串转换为整数

1.仅反转字母

题目链接:917.仅仅反转字母
题目描述在这里插入图片描述

首先,这道题仅仅需要翻转字母,我们先写一个函数来判断是否为字母

bool Isletter(char ch){if (ch >= 'a' && ch <= 'z' || ch>='A' && ch <= 'Z')return true;elsereturn false;}

接着,创建两个索引,beginend,一个从前往后找,找到一个字母停止,另一个从后面找,找到字母停止,然后进行交换,保证begin<end,比较简单,代码如下:

class Solution {
public:bool Isletter(char ch){if (ch >= 'a' && ch <= 'z' || ch>='A' && ch <= 'Z')return true;elsereturn false;}string reverseOnlyLetters(string s){size_t begin1 = 0;size_t  end1 = s.size() - 1;while (begin1 < end1){while (begin1 < end1 && !Isletter(s[begin1])){begin1++;}while (begin1 < end1 && !Isletter(s[end1])){end1--;}swap(s[begin1],s[end1]);begin1++;end1--;}return s;}
};

这里我们直接用了算法库中的swap函数,进行字符的交换

2.字符串中第一个唯一字符

题目链接:387.字符串中第一个唯一字符
题目描述在这里插入图片描述

这道题主要目的就是找第一个唯一出现的字符我们的思路就是类似于计数排序,构建一个存储字符出现次数的数组,最后按照新数组寻找出现一次的那个字符

class Solution {
public:int firstUniqChar(string s) {int arr[256]={0};for(int i=0;i<s.size();i++){arr[s[i]]+=1;}for(int i=0;i<s.size();i++){if(arr[s[i]]==1)return i;}return -1;}
};

解法简单,希望能够理解

3.验证回文串

题目链接:125.验证回文串
题目描述在这里插入图片描述

题目描述,去掉非字母和非数字后的字符串,回文,则构成回文,我们的思路是先判断是否为字母字符或者数字字符:

 bool isLetterOrDigit(char ch) {return (ch >= 'a' && ch <= 'z') || (ch >= 'A' && ch <= 'Z') || (ch >= '0' && ch <= '9');}

接着我们设置两个索引,left,right来从左往右一次寻找字母,左边找到字符则停止,右边找到字符则停止,然后通过字符函数tolower使他们均变为小写字母进行比较

如果有一组不匹配,则返回false
代码如下:

class Solution {
public:bool isLetterOrDigit(char ch) {return (ch >= 'a' && ch <= 'z') || (ch >= 'A' && ch <= 'Z') || (ch >= '0' && ch <= '9');}bool isPalindrome(string s) {int left = 0;int right = s.size() - 1;while (left < right) {while (left < right && !isLetterOrDigit(s[left])) {left++;}while (left < right && !isLetterOrDigit(s[right])) {right--;}if (tolower(s[left]) != tolower(s[right])) {return false;}left++;right--;}return true;}
};

4.字符串相加

题目链接:415.字符串相加
题目描述在这里插入图片描述

本题核心思想就是处理进位问题,从尾部依次相加,结果保留个位数与进位数(0或1),这个进位数进行下一次运算,保留的个位数以新的字符头插在字符串中

class Solution {
public:string addStrings(string num1, string num2) {string result;int end1=num1.size()-1,end2=num2.size()-1;int next=0;while(end1>=0||end2>=0){int val1=end1>=0?num1[end1--]-'0':0;int val2=end2>=0?num2[end2--]-'0':0;int ret=val1+val2+next;next=ret/10;ret=ret%10;result.insert(0,1,ret+'0');}if(next==1){result.insert(0,1,'1');}return result;}
};

函数逻辑

  1. 定义一个空字符串 result,它最终将存储相加后的结果

  2. 定义两个整型变量 end1end2,分别表示 num1num2 字符串的末位索引

  3. 定义变量 next,表示在每一步相加中可能产生的进位

  4. 使用一个 while 循环,条件是 end1end2 中有一个或两个大于或等于0。这表示至少还有一个数字字符串有未处理的数字

  5. 在循环内部,分别计算 val1val2,它们代表当前要相加的两个字符对应的数字值。如果索引小于0,则表示该数字字符串没有更多的位数可以处理,因此对应的值为0

  6. 计算 ret,它是 val1val2 和前一步的进位 next 之和

  7. 更新 nextret 除以10,因为手写加法中,超过10的部分产生进位

  8. 更新 retret 对10取余,因为余数是当前位上的数字

  9. ret 转换为对应的字符,然后将该字符插入 result 字符串的最前面

  10. 重复步骤5-9,直到处理完 num1num2 的所有位数

  11. 循环结束后,检查是否还有未添加的进位 next。如果 next 为1,将字符 ‘1’ 插入 result 字符串的最前面(这一部分可以查阅函数库来了解insert的多种实现形式)

  12. 返回结果字符串 result

优化:
因为不断的头插,数据会挪动,使函数的时间复杂度来到了O(N2)

优化思路:

  • 尾插,再进行反转
class Solution {
public:string addStrings(string num1, string num2) {string result;int end1=num1.size()-1,end2=num2.size()-1;int next=0;while(end1>=0||end2>=0){int val1=end1>=0?num1[end1--]-'0':0;int val2=end2>=0?num2[end2--]-'0':0;int ret=val1+val2+next;next=ret/10;ret=ret%10;result+=ret+'0';}if(next==1){result+='1';}reverse(result.begin(), result.end());return result;}
};

5.反转字符串I I

题目链接:541.反转字符串I I
题目描述在这里插入图片描述

这段代码意在实现根据指定的整数 k 来部分反转字符串 s 的功能。但是,代码中有几个问题需要解决才能正确实现这一功能。

首先,让我们明确正确的逻辑:

  1. 遍历字符串,步长为 2k 字符。
  2. 在每个步长内:
    • 如果剩余字符少于 k 个,则反转这些字符。
    • 如果剩余字符小于 2k 但至少有 k 个,则只反转前 k 个字符
    • 如果有足够的字符,则反转前 k 个字符,保持其余字符不变
class Solution {
public:string reverseStr(string s, int k) {int size = s.size();for (int start = 0; start < size; start += 2 * k) {// 如果剩余字符少于 k 个,则反转剩余的所有字符if (start + k > size) {reverse(s.begin() + start, s.end());} else {// 反转从 start 开始的 k 个字符reverse(s.begin() + start, s.begin() + start + k);// 其余字符保持原样,根据题目要求这部分不需要任何操作}}return s;}
};
  1. 使用一个 for 循环,步长为 2 * k,遍历字符串 s,每次移动2k步,检查并反转前k个字符

  2. 在循环中检查剩余字符的数目,根据这个数目适当地反转字符串的一部分

  3. 使用 reverse 方法来反转从 start 开始的字符。注意,reverse 方法的第二个参数是我们想要反转区间的结束位置的下一个迭代器。如果剩余字符少于 k 个,则 reverse 的参数是 s.end(),这样可以反转从 start 开始的所有剩余字符

  4. 如果 start + k 小于或等于 size,则只反转前 k 个字符,而其余字符保持原样

6.反转字符串中的单词III

题目链接:557.反转字符串中的单词III
题目描述在这里插入图片描述

这道题主要思路就是找到每个空格位置对单词进行分割,逐个翻转

class Solution {
public:string reverseWords(string s) {int size = s.size();size_t pos = s.find(' ');int start = 0;while (pos != string::npos) {reverse(s.begin() + start, s.begin() + pos); start = pos + 1;pos = s.find(' ', start);}reverse(s.begin() + start, s.end()); return s; }
};

注意,reverse接收的参数应该是迭代器,我们应该使用 s.begin() 加上相应的索引来获取正确的迭代器位置,每次找到一个空格就更新索引往后寻找,直到找到最后一个单词结束,结束后,再对最后一个单词进行反转

7.字符串相乘

题目链接:43.字符串相乘
题目描述在这里插入图片描述

思路一:

这道题与我们的字符串相加类似,我们需要做的就是用num2的每一位与num1相乘,每得到两个结果就进行字符串相加

class Solution {
public:string multiply(string num1, string num2) {if (num1 == "0" || num2 == "0") return "0"; string result(num1.size() + num2.size(), '0');for (int i = num1.size() - 1; i >= 0; i--) {for (int j = num2.size() - 1; j >= 0; j--) {int mul = (num1[i] - '0') * (num2[j] - '0');int sum = (result[i + j + 1] - '0') + mul;result[i + j + 1] = (sum % 10) + '0';result[i + j] += sum / 10;}}size_t startpos = result.find_first_not_of("0");if (startpos != string::npos) {return result.substr(startpos);}return "0";}
};

这段代码理解起来十分简单,详细讲解:

此代码模拟了我们在纸上作乘法运算时的过程:

  1. 处理特殊情况
    如果 num1num2 中任意一个是 “0”,那么乘积为 “0”
if (num1 == "0" || num2 == "0") return "0";
  1. 初始化结果字符串
    初始化结果字符串 result,长度为 num1.size() 加上 num2.size(),所以 result 的长度足以存储乘法得到的所有可能数字,包括合并进位。所有字符先被设置为 '0'
string result(num1.size() + num2.size(), '0');
  1. 嵌套循环
    外层循环以 num1.size() - 1 开始,即 num1 字符串的最后一个字符,向前遍历整个字符串。内层循环以 num2.size() - 1 开始,即 num2 字符串的最后一个字符,向前遍历整个字符串。循环索引 ij 分别对应 num1num2 的每个数位
for (int i = num1.size() - 1; i >= 0; i--) {for (int j = num2.size() - 1; j >= 0; j--) {// ...}
}
  1. 计算乘积
    对于每对数位(num1[i], num2[j]),计算它们值的乘积,另外再把结果 result 相应位置上的值加上去(需要先把result[i+j+1]字符转化为整数)。需要注意的是,计算中还会加上之前的进位
int mul = (num1[i] - '0') * (num2[j] - '0');
int sum = (result[i + j + 1] - '0') + mul;
  1. 处理结果和进位
    当前乘积mul与结果result当前位置上的数相加后,可能会大于等于10,即产生进位。将当前位的结果模上10得到最终结果,并把商加到下一位上
result[i + j + 1] = (sum % 10) + '0';
result[i + j] += sum / 10;
  1. 移除前导零
    由于乘积可能比 num1.size() + num2.size() 这么长要短,所以会在 result 最前面出现一些 ‘0’。我们需要找到第一个不是 ‘0’ 的字符的位置,并从那里开始返回子字符串
size_t startpos = result.find_first_not_of("0");
if (startpos != string::npos) {return result.substr(startpos);
}

如果整个字符串都是 ‘0’,说明两个数的乘积是 0,直接返回 “0”。

  1. 返回结果
    如果 result 中有非 ‘0’ 的字符,就从第一个非零字符开始返回剩余的子字符串,否则直接返回 “0”。

8.把字符串转换为整数

题目链接:LCR 192.把字符串转换为整数(atoi)
题目描述在这里插入图片描述

首先,我们写两个函数来对空字符和正负号进行处理:

int i = 0;
int sign = 1;
int result = 0;
while (i < str.length() && str[i] == ' ') { i++;}if (i < str.length() && (str[i] == '+' || str[i] == '-')) {sign = (str[i] == '+') ? 1 : -1;i++;}

接着处理数字部分,如果超过32 位有符号整数范围,则进行截断:

class Solution {
public:int myAtoi(string str) {int i = 0;int sign = 1;int result = 0;while (i < str.size() && str[i] == ' ') { i++;}if (i < str.size() && (str[i] == '+' || str[i] == '-')) {sign = (str[i] == '+') ? 1 : -1;i++;}while (i < str.size() && isdigit(str[i])) { int digit = str[i] - '0';if (result > INT_MAX / 10 || (result == INT_MAX / 10 && digit > INT_MAX % 10)) {return sign == 1 ? INT_MAX : INT_MIN;}result = 10 * result + digit;i++;}return result * sign;}
};
 if (result > INT_MAX / 10 || (result == INT_MAX / 10 && digit > INT_MAX % 10)) {return sign == 1 ? INT_MAX : INT_MIN;}

这部分代码的目的是检查在将下一个数字添加到已解析的结果 result 之前,是否会导致整数溢出。溢出指的是整数的值超出了它能表示的最大范围。在C++中,对于32位 int 类型,能够表示的最大整数值定义在 <climits> 头文件中,称为 INT_MAX,通常为 2^31 - 1(即2147483647),最小整数值为 INT_MIN,通常为 -2^31(即-2147483648)
为了避免result在由字符串转换为整数时溢出,代码使用了下列条件检查:

  1. result > INT_MAX / 10
    这个检查确保将当前的 result 乘以 10(也就是添加新的数字之前)不会超过最大整数值 INT_MAX。如果 result 已经大于 INT_MAX 除以 10,那么在下一步乘以 10 时一定会发生溢出。

  2. (result == INT_MAX / 10 && digit > INT_MAX % 10)
    如果 result 已经达到了 INT_MAX 除以 10 的值,那么我们可以检查下一个要添加的数字(digit)是否会导致溢出。因为我们知道 result 乘以 10 刚好达到但不超过 INT_MAX,所以我们只需要保证添加的数字小于或等于 INT_MAX 最后一个数位的数字的值,即 INT_MAX % 10。如果 digit 大于这个值,那么加上 digit 之后会超出 INT_MAX,发生溢出

如果以上任何一种溢出条件满足,那么根据数字的正负符号,函数返回最大或最小的 int 值:

return sign == 1 ? INT_MAX : INT_MIN;

  • sign 为 1,即正数的情况下,返回 INT_MAX
  • sign 为 -1,即负数的情况下,返回 INT_MIN

本节内容到此结束,感谢大家阅读,可以多多交流!!

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

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

相关文章

一篇文章带您了解面向对象(java)

1.简单理解面向过程编程和面向对象编程 面向过程编程&#xff1a;开发一个一个的方法&#xff0c;有数据需要处理&#xff0c;我们就可以调用方法来处理。 package com.web.quictstart;public class demo1 {public static void main(String[] args) {totalScore("张三&q…

mac上VMware fusion net模式无法正常使用的问题

更新时间&#xff1a;2024年04月22日21:39:04 1. 问题 环境&#xff1a; intel芯片的macbook pro VMware fusion 13.5.1 无法将“Ethernet0”连接到虚拟网络“/dev/vmnet8”。在这里显示这个之后&#xff0c;应该是vmnet8的网段发生了冲突&#xff0c;所以导致无法正常使用…

前端开发攻略---拖动归类,将元素拖拽到相应位置

1、演示 2、代码 <!DOCTYPE html><html lang"en"><head><meta charset"UTF-8" /><meta http-equiv"X-UA-Compatible" content"IEedge" /><meta name"viewport" content"widthdevice-…

2024年Q1季度平板电视行业线上市场销售数据分析

Q1季度平板电视线上市场表现不如预期。 根据鲸参谋数据显示&#xff0c;2024年1月至3月线上电商平台&#xff08;京东天猫淘宝&#xff09;平板电视累计销量约360万件&#xff0c;环比下降12%&#xff0c;同比下降30%&#xff1b;累计销售额约99亿元&#xff0c;环比下降28%&a…

学习STM32第十七天

备份域详解 一、简介 在参考手册的电源控制章节&#xff0c;提到了备份域&#xff0c;BKPR是在RTC外设中用到&#xff0c;包含20个备份数据寄存器&#xff08;80字节&#xff09;&#xff0c;备份域包括4KB的备份SRAM&#xff0c;以32位、16位或8位模式寻址&#xff0c;在VBAT…

C++初阶学习第二弹——C++入门(下)

C入门&#xff08;上&#xff09;&#xff1a;C初阶学习第一弹——C入门&#xff08;上&#xff09;-CSDN博客 目录 一、引用 1.1 引用的实质 1.2 引用的用法 二、函数重载 三、内敛函数 四、auto关键字 五、总结 前言&#xff1a; 在上面一章我们已经讲解了C的一些基本…

Vue2进阶之Vue2高级用法

Vue2高级用法 mixin示例一示例二 plugin插件自定义指令vue-element-admin slot插槽filter过滤器 mixin 示例一 App.vue <template><div id"app"></div> </template><script> const mixin2{created(){console.log("mixin creat…

【Java网络编程】TCP通信(Socket 与 ServerSocket)和UDP通信的三种数据传输方式

目录 1、TCP通信 1.1、Socket 和 ServerSocket 1.3、TCP通信示例 2、UDP的三种通信&#xff08;数据传输&#xff09;方式 1、TCP通信 TCP通信协议是一种可靠的网络协议&#xff0c;它在通信的两端各建立一个Socket对象 通信之前要保证连接已经建立&#xff08;注意TCP是一…

【Interconnection Networks 互连网络】Torus 网络拓扑

1. Torus 网络拓扑2. Torus 网络拓扑结构References 1. Torus 网络拓扑 Torus 和 Mesh 网络拓扑&#xff0c;又可以称为 k-ary n-cubes&#xff0c;在规则的 n 维网格中包裹着 N k^n 个节点&#xff0c;每个维度都有 k 个节点&#xff0c;并且最近邻居之间有通道。k-ary n-c…

YOLOv9有效改进专栏汇总|未来更新卷积、主干、检测头注意力机制、特征融合方式等创新![2024/4/21]

​ 专栏介绍&#xff1a;YOLOv9改进系列 | 包含深度学习最新创新&#xff0c;助力高效涨点&#xff01;&#xff01;&#xff01; 专栏介绍 YOLOv9作为最新的YOLO系列模型&#xff0c;对于做目标检测的同学是必不可少的。本专栏将针对2024年最新推出的YOLOv9检测模型&#xff0…

《HCIP-openEuler实验指导手册》1.3Apache动态功能模块加载卸载练习

1.3.1 配置思路 mod_status 模块可以帮助管理员通过web界面监控Apache运行状态&#xff0c;通过LoadModule指令加载该模块&#xff0c;再配置相关权限&#xff0c;并开启ExtendedStatus后&#xff0c;即可使用该模块。 1.3.2 配置步骤 检查mod_status模块状态&#xff08;使…

net模块

建立TCP的链接 1 发送消息的服务 2 接收消息 2 建立http的链接让浏览器进行访问 import net from netconst html <h1>TCP</h1>const respinseHeaders [HTTP/1.1 200 OK,Content-Type:text/html,Content-Length: html.length,\r\n,html]const http net.create…

RK3568 学习笔记 : u-boot 通过 tftp 网络更新 u-boot自身

前言 开发板型号&#xff1a; 【正点原子】 的 RK3568 开发板 AtomPi-CA1 使用 虚拟机 ubuntu 20.04 收到单独 编译 RK3568 u-boot 使用 rockchip Linux 内核的设备树 【替换】 u-boot 下的 rk3568 开发板设备树文件&#xff0c;解决 u-boot 下千兆网卡设备能识别但是无法 Pi…

Spring(下)

接上篇&#xff0c;从第八个问题讲起 八.Spring工厂创建复杂对象 1.什么是复杂对象 简单对象就是可以直接new出来的&#xff0c;也就是直接调用构造方法创建 所以复杂对象就是不能直接通过调用构造方法创建。就比如JDBC中的Connection 2.三种方法 &#xff08;1&#xff…

【华为 ICT HCIA eNSP 习题汇总】——题目集18

1、SSH默认工作使用的TCP端口号是&#xff08;&#xff09;。 A、20 B、21 C、22 D、23 考点&#xff1a;①传输层 ②应用层 解析&#xff1a;&#xff08;C&#xff09; SSH为建立在应用层和传输层上的安全协议&#xff0c;是对TCP/IP协议的传输层以上的SSH会话流程进行加密的…

必应搜索广告与谷歌搜索广告对比那个更好?

搜索引擎广告作为企业获取潜在客户的重要渠道之一&#xff0c;其效果直接关系到营销策略的成功与否。两大搜索引擎巨头——谷歌&#xff08;Google&#xff09;和必应&#xff08;Bing&#xff09;各自提供了广告平台&#xff0c;即谷歌广告&#xff08;Google Ads&#xff09;…

【Web】2022DASCTF X SU 三月春季挑战赛 题解(全)

目录 ezpop calc upgdstore ezpop 瞪眼看链子 fin#__destruct -> what#__toString -> fin.run() -> crow#__invoke -> fin#__call -> mix.get_flag() exp <?php class crow {public $v1;public $v2;}class fin {public $f1; }class what {public $a; }…

力扣hot100(python解析)

文章目录 13. 合并区间14. 轮转数组 13. 合并区间 合并区间 数组 排序 思路&#xff1a; 代码&#xff1a; class Solution:def merge(self, intervals):result []if len(intervals) 0:return result # 区间集合为空直接返回intervals.sort() # 默认按照区间的左边界进…

z3求解器脚本(CTF-reverse必备)

CTF-reverse中有一类题目是通过约束方程求解变量的值&#xff0c;然后转化为对应的ASCII码&#xff0c;最终获得flag&#xff0c;约束方程以及要求解的未知数往往非常多&#xff0c;因此手算十分不现实&#xff0c;借助python中的z3模块可以很快完成求解&#xff01; 下面是某道…

开始Java之旅

1.Java语言 java是一门优秀的程序设计语言&#xff0c;并且是一种半编译型&#xff0c;半解释型语言。 Java 语言源于 1991 年 4 月&#xff0c;Sun 公司 James Gosling博士 领导的绿色计划(Green Project) 开始启动&#xff0c;此计划最初的目标是开发一种能够在各种消费性电…