string类常见题目详解(二) —— 仅仅反转字母、字符串中的第一个唯一字母、字符串最后一个单词的长度、验证回文串、字符串相加

本文主要是介绍string类常见题目详解(二) —— 仅仅反转字母、字符串中的第一个唯一字母、字符串最后一个单词的长度、验证回文串、字符串相加,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本专栏内容为:C++学习专栏,分为初阶和进阶两部分。 通过本专栏的深入学习,你可以了解并掌握C++。

💓博主csdn个人主页:小小unicorn
⏩专栏分类:C++
🚚代码仓库:小小unicorn的代码仓库🚚
🌹🌹🌹关注我带你学习编程知识

目录

  • 题目一:仅仅反转字母
    • 题目描述:
    • 示例:
    • 思路:
    • 代码
  • 题目二:字符串中的第一个唯一字母
    • 题目描述:
    • 示例:
    • 思路:
    • 代码:
  • 题目三:字符串最后一个单词的长度
    • 题目描述:
    • 示例:
    • 思路:
    • 重点:
    • 代码:
  • 题目四:验证回文串
    • 题目描述:
    • 示例:
    • 思路:
    • 代码:
  • 题目五:字符串相加
    • 题目描述:
    • 示例:
    • 思路:
    • 代码:
      • 优化:
      • 优化后的代码:

题目一:仅仅反转字母

题目描述:

给定一个字符串,返回“反转后的”字符串,其中不是字母的字符都保留在原地,而所有字母的位置发生反转。

示例:

输入:“a-bC-dEf-ghIj”
 输出:“j-Ih-gfE-dCba”

思路:

使用两个指针,开始时头指针指向字符串开头,尾指针指向字符串末尾(’\0’的前一个字符)。头指针先向后寻找待反转字母,尾指针再向前寻找待反转字母,然后反转这两个待反转字母即可。重复该步骤,直到头指针和尾指针发生错位为止。

动图演示:
在这里插入图片描述

代码

class Solution 
{
public://判断字符ch是否是字母bool IsLetter(char ch){if ((ch >= 'a'&&ch <= 'z') || (ch >= 'A'&&ch <= 'Z'))return true;elsereturn false;}//仅仅反转字母string reverseOnlyLetters(string s) {if (s.empty()) //若s为空字符串,则返回其本身return s;size_t begin = 0, end = s.size() - 1; //定义头指针和尾指针,用于反转字母while (begin < end) //当还有字母可反转时,循环继续{while (begin < end && !IsLetter(s[begin])) //头指针寻找待反转字母begin++;while (begin < end && !IsLetter(s[end])) //尾指针寻找待反转字母end--;swap(s[begin], s[end]); //交换这两个待反转字母begin++;end--;}return s; //返回反转后的字符串}
};

注意:若传入的字符串尾空字符串,则直接返回即可。

题目二:字符串中的第一个唯一字母

题目描述:

给定一个字符串,找到它的第一个不重复的字母,并返回它的索引。如果不存在,则返回-1。

示例:

s = “loveleetcode”
 返回 2

思路:

用一个含义26个元素的数组统计26个字母在字符串中出现的次数,然后再遍历一次字符串,寻找字符串中第一个只出现一次的字母,并返回它的索引,若不存在,则返回-1。

代码:

class Solution 
{
public:int firstUniqChar(string s) {size_t table[26] = { 0 };//统计26个字母在字符串中出现的次数for (size_t i = 0; i < s.size(); i++){table[s[i] - 'a']++;}//寻找字符串中的第一个只出现一次字母for (size_t i = 0; i < s.size(); i++){if (table[s[i] - 'a'] == 1)return i; //返回下标索引}return -1; //不存在,返回-1}
};

题目三:字符串最后一个单词的长度

题目描述:

计算字符串最后一个单词的长度,单词以空格隔开,字符串长度小于5000。

示例:

输入:“hello CSDN”
 输出:4

思路:

先找到字符串中最后一个空格的位置,空格之后的字符个数就是最后一个单词的长度。若字符串中不存在空格,则字符串的长度就是最后一个单词的长度。

重点:

该题的思路非常容易想到,但该题真正的难点不是如何求得字符串中最后一个单词的长度,而是如何读取一个含有空格的字符串。因为操作符>>读取到空格便会停止,所以不能使用>>读取目标字符串,这时我们就需要用到getline函数了,该函数便可以读取含有空格的内容。

代码:

#include <iostream>
#include <string>
using namespace std;
int main()
{string s;getline(cin, s); //从cin读取一行含有空格的字符串size_t pos = s.rfind(' '); //获取字符串中最后一个空格的位置if (pos == string::npos) //字符串中不含空格{//输出字符串的长度cout << s.size() << endl;}else //字符串中含有空格{//输出字符串的长度 - 最后一次出现空格的位置 - 1cout << s.size() - pos - 1 << endl;}return 0;
}

题目四:验证回文串

题目描述:

给定一个字符串,验证它是否是回文串,只考虑字母和数字字符,可以忽略字母的大小写。

示例:

输入:“A man, a plan, a canal: Panama”
 输出:true

思路:

题目中说到可以忽略字母大小写,即’a’可与’a’匹配,‘a’还可与’A’匹配,所以我们可以先将字符串中所有的大写字母转换为小写字母,然后再进行进一步判断。
 判断时使用两个指针,开始时头指针指向字符串开头,尾指针指向字符串末尾(’\0’的前一个字符)。头指针先向后寻找待判断的字母或数字字符,尾指针再向前寻找待判断的字母或数字字符,然后判断这两个字符是否相等,若相等,则继续下一次判断;若不相等,则该字符串不是回文串。重复该步骤,直到头指针和尾指针发生错位为止,此时便可确定该字符串是回文串。

代码:

class Solution 
{
public://判断ch是否是小写字母或数字字符bool isLetterOrNumber(char ch){if ((ch >= 'a' && ch <= 'z')|| (ch >= '0' && ch <= '9'))return true;elsereturn false;}//验证回文串bool isPalindrome(string s) {//将字符串中所有大写字母转为小写字母for (auto& ch : s){if (ch >= 'A' && ch <= 'Z')ch += 32;}int begin = 0, end = s.size() - 1; //定义头指针和尾指针,用于验证回文串while (begin < end) //当还有字母未判断时,循环继续{while (begin < end && !isLetterOrNumber(s[begin])) //头指针寻找待判断的字母或数字字符begin++;while (begin < end && !isLetterOrNumber(s[end])) //尾指针寻找待判断的字母或数字字符end--;if (s[begin] == s[end]) //待判断字符相等,继续下一次判断{begin++;end--;}else{return false; //待判断字符不相等,该字符串不是回文串}}return true; //字符串中所有字符判断完毕,该字符串是回文串}
};

题目五:字符串相加

题目描述:

给定两个字符串形式的非负整数num1和num2,计算它们的和。

示例:

输入:“999”, “1”
 输出:“1000”

思路:

和我们平时计算两个数的和一样,我们从两个字符串的最后一个数字开始进行相加,并设置一个变量记录是否需要进位,这样一来,两个字符串相加的时候,每一位置的数字 = 字符串1对应位置的数字 + 字符串2对应位置的数字 + 进位变量,若相加后该位置的数字大于9,则说明需要进位,这时我们设置进位变量为1(两个数字相加,最多只能进位1),并将该位置的数字减去10后的结果作为相加后该位置的数字即可,如此进行下去,直到两个字符串都遍历完毕即可。特别注意,两个字符串相加结束后还需要判断进位变量是否为1,若为1,则需要头插一个字符1到最终的字符串中。

代码:

class Solution 
{
public:string addStrings(string num1, string num2) {int end1 = num1.size() - 1, end2 = num2.size() - 1; //定义两个字符串的尾指针string RetStr; //存储两个字符串相加后的结果int next = 0; //标识进位while (end1 >= 0 || end2 >= 0) //两个字符串中有一个未遍历完,则继续循环{int val1 = 0; //第一个字符串等待相加的数字if (end1 >= 0){val1 = num1[end1] - '0';end1--;}int val2 = 0; //第二个字符串等待相加的数字if (end2 >= 0){val2 = num2[end2] - '0';end2--;}int RetVal = val1 + val2 + next; //两个数字相加后的结果,注意需要加上进位if (RetVal > 9) //判断是否需要进位{RetVal -= 10;next = 1; //需要进位,设置next为1}else{next = 0; //不需进位,设置next为0}RetStr.insert(0, 1, RetVal + '0'); //将RetVal头插到RetStr}if (next == 1) //判断是否还需进位RetStr.insert(0, 1, '1'); //将'1'头插到RetStrreturn RetStr; //返回这两个字符串相加后的结果}
};

优化:

代码中,我们每得到一个位置(个位、十位、百位…)的结果就需要头插一个数字到最终的字符串中。而我们知道,头插的代价是比较大的,因为我们每次进行头插就需要将所有的数据都向后挪动一位,留出最前面的位置以供插入,这种算法的时间复杂度达到O ( N ) O(N)O(N)。
 解决这个问题也不难,我们可以将得到的每一位数字都尾插到字符串后面,只需最后进行一次字符串反转即可。

优化后的代码:

class Solution 
{
public:string addStrings(string num1, string num2) {int end1 = num1.size() - 1, end2 = num2.size() - 1; //定义两个字符串的尾指针string RetStr; //存储两个字符串相加后的结果int next = 0; //标识进位while (end1 >= 0 || end2 >= 0) //两个字符串中有一个未遍历完,则继续循环{int val1 = 0; //第一个字符串等待相加的数字if (end1 >= 0){val1 = num1[end1] - '0';end1--;}int val2 = 0; //第二个字符串等待相加的数字if (end2 >= 0){val2 = num2[end2] - '0';end2--;}int RetVal = val1 + val2 + next; //两个数字相加后的结果,注意需要加上进位if (RetVal > 9) //判断是否需要进位{RetVal -= 10;next = 1; //需要进位,设置next为1}else{next = 0; //不需进位,设置next为0}RetStr += (RetVal + '0'); //将RetVal尾插到RetStr}if (next == 1) //判断是否还需进位RetStr += '1'; //将'1'尾插插到RetStrreverse(RetStr.begin(), RetStr.end()); //将字符串RetStr进行反转return RetStr; //返回这两个字符串相加后的结果}
};

这篇关于string类常见题目详解(二) —— 仅仅反转字母、字符串中的第一个唯一字母、字符串最后一个单词的长度、验证回文串、字符串相加的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java实现字符串大小写转换的常用方法

《Java实现字符串大小写转换的常用方法》在Java中,字符串大小写转换是文本处理的核心操作之一,Java提供了多种灵活的方式来实现大小写转换,适用于不同场景和需求,本文将全面解析大小写转换的各种方法... 目录前言核心转换方法1.String类的基础方法2. 考虑区域设置的转换3. 字符级别的转换高级转换

HTML5的input标签的`type`属性值详解和代码示例

《HTML5的input标签的`type`属性值详解和代码示例》HTML5的`input`标签提供了多种`type`属性值,用于创建不同类型的输入控件,满足用户输入的多样化需求,从文本输入、密码输入、... 目录一、引言二、文本类输入类型2.1 text2.2 password2.3 textarea(严格

Python数据验证神器Pydantic库的使用和实践中的避坑指南

《Python数据验证神器Pydantic库的使用和实践中的避坑指南》Pydantic是一个用于数据验证和设置的库,可以显著简化API接口开发,文章通过一个实际案例,展示了Pydantic如何在生产环... 目录1️⃣ 崩溃时刻:当你的API接口又双叒崩了!2️⃣ 神兵天降:3行代码解决验证难题3️⃣ 深度

C++ move 的作用详解及陷阱最佳实践

《C++move的作用详解及陷阱最佳实践》文章详细介绍了C++中的`std::move`函数的作用,包括为什么需要它、它的本质、典型使用场景、以及一些常见陷阱和最佳实践,感兴趣的朋友跟随小编一起看... 目录C++ move 的作用详解一、一句话总结二、为什么需要 move?C++98/03 的痛点⚡C++

MySQL字符串转数值的方法全解析

《MySQL字符串转数值的方法全解析》在MySQL开发中,字符串与数值的转换是高频操作,本文从隐式转换原理、显式转换方法、典型场景案例、风险防控四个维度系统梳理,助您精准掌握这一核心技能,需要的朋友可... 目录一、隐式转换:自动但需警惕的&ld编程quo;双刃剑”二、显式转换:三大核心方法详解三、典型场景

MySQL中between and的基本用法、范围查询示例详解

《MySQL中betweenand的基本用法、范围查询示例详解》BETWEENAND操作符在MySQL中用于选择在两个值之间的数据,包括边界值,它支持数值和日期类型,示例展示了如何使用BETWEEN... 目录一、between and语法二、使用示例2.1、betwphpeen and数值查询2.2、be

python中的flask_sqlalchemy的使用及示例详解

《python中的flask_sqlalchemy的使用及示例详解》文章主要介绍了在使用SQLAlchemy创建模型实例时,通过元类动态创建实例的方式,并说明了如何在实例化时执行__init__方法,... 目录@orm.reconstructorSQLAlchemy的回滚关联其他模型数据库基本操作将数据添

Java中ArrayList与顺序表示例详解

《Java中ArrayList与顺序表示例详解》顺序表是在计算机内存中以数组的形式保存的线性表,是指用一组地址连续的存储单元依次存储数据元素的线性结构,:本文主要介绍Java中ArrayList与... 目录前言一、Java集合框架核心接口与分类ArrayList二、顺序表数据结构中的顺序表三、常用代码手动

JAVA线程的周期及调度机制详解

《JAVA线程的周期及调度机制详解》Java线程的生命周期包括NEW、RUNNABLE、BLOCKED、WAITING、TIMED_WAITING和TERMINATED,线程调度依赖操作系统,采用抢占... 目录Java线程的生命周期线程状态转换示例代码JAVA线程调度机制优先级设置示例注意事项JAVA线程

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra