【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;
		else
			return false;
	}

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

class Solution {
public:
	bool Isletter(char ch)
	{
		if (ch >= 'a' && ch <= 'z' || ch>='A' && ch <= 'Z')
			return true;
		else
			return 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

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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C++:模板详解

模板详解 1.函数模板1.概念2.语法3.原理4.实例化1.隐式实例化2.显示实例化 5.匹配原则 2.类模板1.格式2.实例化 3.非类型模板参数注意点 4.特化1.概念2.函数模板特化1.前提2.语法说明3.示例 3.类模板特化1.全特化2.偏特化/半特化3.选择顺序 4.按需实例化 5.模板的分离编译1.分离…

开发一个农场小游戏需要多少钱

开发一个农场小游戏的费用因多个因素而异&#xff0c;包括但不限于游戏的规模、复杂性、功能需求、设计复杂度、开发团队的规模和经验&#xff0c;以及项目的时间周期等。因此&#xff0c;无法给出确切的费用数字。 具体来说&#xff0c;游戏的复杂程度和包含的功能特性数量会直…

巧用断点设置查找bug【debug】

默认设置的断点&#xff0c;当代码运行到断点处MCU就会被挂起&#xff0c;从而停在断点处。 但在某些情况下&#xff0c;如调试FCCU时&#xff0c;如果设置断点&#xff0c;MCU停下后将会导致 FCCU 配置WDG超时。或在调试类似电机控制类的应用时&#xff0c;不适当的断点会导 致…

镜舟科技荣获金科创新社 2024 年度金融数据智能解决方案奖

近日&#xff0c; 镜舟科技凭借领先的金融实时数仓构建智能经营解决方案&#xff0c;在“金科创新社第六届金融数据智能优秀解决方案评选”活动中&#xff0c;成功入选“数据治理与数据平台创新优秀解决方案”榜单。 金科创新社主办的“鑫智奖”评选活动&#xff0c;旨在展示…

详解IIC通信协议以及FPGA实现

一、IIC简介 IIC也称为I2C&#xff08;Inter-Integrated Circuit&#xff09;由飞利浦公司&#xff08;现在的恩智浦半导体&#xff09;开发&#xff0c;是一种用于短距离数字通信的串行&#xff0c;同步&#xff0c;半双工通信接口协议&#xff1b;传输在标准模式下可以达到10…

python:元组,字符串,切片

一、元组# 列表可以修改内容&#xff0c;元组可以不被修改 # 在程序内封装数据&#xff0c;不希望数据被篡改&#xff0c;所以使用元组 # 语法&#xff1a; 不限制类型 # 定于元组的字面量&#xff1a; &#xff08;元素&#xff0c;元素&#xff0c;元素.....&#xff09; # 定…

apipost、postman等工具上传图片测试flask、fastapi的文件api接口

参考&#xff1a;https://blog.csdn.net/qq_15821487/article/details/119354129 https://www.cnblogs.com/wyxjava/p/16076176.html 选择from-data&#xff0c;下拉选择file上传文件发送即可

线上真实案例之执行一段逻辑后报错Communications link failure

1.问题发现 在开发某个项目的一个定时任务计算经销商返利的功能时&#xff0c;有一个返利监测的调度&#xff0c;如果某一天返利计算调度失败了&#xff0c;需要重新计算&#xff0c;这个监测的调度就会重新计算某天的数据。 在UAT测试通过&#xff0c;发布生产后&#xff0c…

NVIDIA安装程序失败-Nsight Visual Studio Edition失败解决办法

博主是要升级cuda版本&#xff0c;那么在安装新版本之前需要卸载以前的版本。 博主一溜卸载下去&#xff0c;最后有这么个东西卸载不掉&#xff0c;Nsight Visual Studio Edition 不管是电脑系统卸载还是360卸载&#xff0c;都卸载不掉。 此时安装新的cuda也遇到了这个问题 由…

PLC存储器分类及西门子SIMATIC S7-1200存储器参数

存储器用来储存程序和数据&#xff0c;分为系统存储器和用户存储器。 系统存储器存放由PLC生产厂商编写好的系统程序&#xff0c;并固化在只读存储器&#xff08;ROM&#xff09;内&#xff0c;用户不能修改。用户存储器存放用户根据控制要求编写的应用程序。目前大多数PLC采用…

面试经典150题——从中序与后序遍历序列构造二叉树

1. 题目描述 2. 题目分析与解析 其实这个题目和昨天那个很相似&#xff0c;思考思路如下&#xff1a; 解决从中序&#xff08;inorder&#xff09;与后序&#xff08;postorder&#xff09;遍历序列构造二叉树的问题时&#xff0c;考虑到这两个遍历序列为我们提供了树结构中…

解决方案 SHUTDOWN_STATE xmlrpclib.py line: 794 ERROR: supervisor shutting down

Supervisor操作命令 重新加载 Supervisor 配置&#xff1a; sudo supervisorctl reread sudo supervisorctl update sudo supervisorctl restart all这将重新读取 Supervisor 的配置文件&#xff0c;更新进程组&#xff0c;然后重启所有进程。 查看 Supervisor 日志&#xff1…

尺取法知识点讲解

一、固定长度的情况&#xff1a; 最小和(sum) 输入N个数的数列&#xff0c;所有相邻的M个数的和共有N-M1个&#xff0c;求其中的最小值。 输入格式 第1行&#xff0c;2个整数N&#xff0c;M&#xff0c;范围在[3…100000]&#xff0c;N>M。 第2行&#xff0c;有N个正…

R语言入门:“Hellinger“转化和“normalize“转化(弦转化)的公式表示与R代码实现

1、写在前面 vegan包中的decostand()函数为群落生态学研究提供了一些流行的(和有效的)标准化方法。有关decostand()函数标准化的一些标准化方法可以看我的另一篇笔记&#xff1a;R语言入门&#xff1a;vegan包使用decostand()函数标准化方法 由于在网络上没有找到关于这两个转…

Redis-键值设计

Redis-键值设计 1.设置key的规范 遵循基本格式&#xff1a;【业务名称】&#xff1a;【数据名】&#xff1a;【id】 可读性强&#xff0c;在客户端的情况下使用:如果前缀相同会分目录层级长度不超过44字节 string数据结构的三种类型&#xff0c;在44字节之内是embstring 内存…

鸿蒙应用开发之Web组件3

前面学习了从网上加载网页的显示,本文将要学习加载本地的网页。比如很多显示的内容,可以制作网页的文件格式,然后直接使用它来显示,就可以减少界面的制作。另外,当手机没有网络的时候,如果想从网络上获取内容就会失败,这时候可以使用本地的网页内容来代替。这样不会导致…

Python的Logging模块高级用法-日志处理

&#x1f47d;发现宝藏 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。【点击进入巨牛的人工智能学习网站】。 探索Python中的日志处理&#xff1a;Logging模块的高级用法 在Python应用程序中&#xff0…

lementui el-menu侧边栏占满高度且不超出视口

做了几次老是忘记&#xff0c;这次整理好逻辑做个笔记方便重复利用&#xff1b; 问题&#xff1a;elementui的侧边栏是占不满高度的&#xff1b;但是使用100vh又会超出视口高度不美观&#xff1b; 解决办法&#xff1a; 1.获取到侧边栏底部到视口顶部的距离 2.获取到视口的高…

【动态规划】dp 路径问题(不同路径、路径最小和、地下城游戏...)

文章目录 1. 前言 - 理解动态规划算法1.5 关于dp路径问题2. 例题2.1_不同路径Warning. 关于状态表示 3. 算法题3.1_不同路径II3.2_珠宝的最高价值3.3_下降路径最小和3.4_最小路径和3.5_地下城游戏关于状态表示的两种选法&#xff1a; 1. 前言 - 理解动态规划算法 关于 动态规划…

Pytorch 之torch.nn初探 池化--Pooling Layers

任务描述 本关任务&#xff1a;本关提供了一个Variable 类型的变量x&#xff0c;要求按照条件创建一个Conv2d变量conv&#xff0c;一个MaxPool2d变量pool&#xff0c;对x应用卷积和最大池化操作并赋值给变量outpout_pool&#xff0c;并输出outpout_pool 的大小。 相关知识 P…