(第13、14天)【leetcode题解】#右旋字符串 28、找出字符串中第一个匹配项的下标

2024-05-09 20:04

本文主要是介绍(第13、14天)【leetcode题解】#右旋字符串 28、找出字符串中第一个匹配项的下标,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

  • 右旋字符串
    • 题目描述
    • 思路
    • 代码
    • 思考
  • 28、找出字符串中第一个匹配项的下标
    • 题目描述
    • 暴力匹配
      • 思路
      • 代码
    • KMP算法
      • 思路
      • 代码
      • 难点回顾

右旋字符串

题目描述

  • 字符串的右旋转操作是把字符串尾部的若干个字符转移到字符串的前面。
  • 给定一个字符串 s 和一个正整数 k,请编写一个函数,将字符串中的后面 k 个字符移到字符串的前面,实现字符串的右旋转操作。
  • 例如,对于输入字符串 “abcdefg” 和整数 2,函数应该将其转换为 “fgabcde”。

思路

  • 思路一:开辟新的空间,拆分原字符串按要求放入新空间。
  • 思路二:不开辟新空间,原地反转。先全部反转,之后局部反转。

本题的解法使用思路二。

代码

版本一

#include<iostream>
#include<algorithm>
using namespace std;int main() {int k;string s;cin >> k;cin >> s;reverse(s.begin(), s.end());//整体反转reverse(s.begin(), s.begin() + k);//反转前一段,反转的长度为kreverse(s.begin() + k, s.end());//反转后一段cout << s << endl;
}

版本二:反转顺序发生改变。先局部反转,再整体反转。

#include<iostream>
#include<algorithm>
using namespace std;int main() {int k;string s;cin >> k;cin >> s;int len = s.size();//获取字符串长度reverse(s.begin(), s.begin() + len - k);//反转前一段reverse(s.begin() + len - k, s.end());//反转后一段reverse(s.begin(), s.end());//整体反转cout << s << endl;
}

时间复杂度:O(n);翻转操作需要遍历数量级为n的次数。
空间复杂度:O(1);原地翻转,没有开辟额外内存空间。

思考

1.字符串的操作和数组有共性:需要下标访问、各种操作都是基于使用下标进行的元素修改。
2. 目前所遇到的关于字符串的题目都是类似于翻转、改变字符和位置等。这都需要使用指针(下标),双指针是常用的,它可以帮助原地修改字符串,还可以降低时间复杂度。

28、找出字符串中第一个匹配项的下标

题目描述

给你两个字符串 haystack 和 needle ,请你在 haystack 字符串中找出 needle 字符串的第一个匹配项的下标(下标从 0 开始)。如果 needle 不是 haystack 的一部分,则返回 -1 。

题目类型为字符串匹配

暴力匹配

思路

  • 将字符串needlehaystack中的所有子串逐个匹配。
  • 为了减少不必要的匹配,一旦匹配失败,立即跳入下一个子串的匹配。

代码

class Solution {
public:int strStr(string haystack, string needle) {int n = haystack.size(), m = needle.size();for (int i = 0; i + m <= n; i++) {//从i开始遍历m个字符,如果从i开始不足m个字符,表明已经没有符合要求的子串bool flag = true;//进行匹配操作for (int j = 0; j < m; j++) {//从0开始遍历子串//如果有一个不相同if (haystack[i + j] != needle[j]) {flag = false;break;//结束本次子串的匹配}}if (flag) {return i;}}return -1;}
};

时间复杂度:O(n*m);最多遍历全部的本文串,并在每次遍历文本串的内部遍历模式串。
空间复杂度:O(1);

KMP算法

思路

  1. 总思路:遍历文本串和模式串,匹配模式串。当遇到字符不匹配的情况时,根据前缀表回退到模式串已经匹配好的位置继续匹配。
  2. 前缀表:长度跟模式串相同,记录每个子串的最长公共前后缀长度。使用最长公共前后缀长度,可以知道回退到哪个位置是已经匹配好的位置,从这个位置开始继续匹配即可,可以提高匹配效率。
  3. 根据前缀表匹配:循环遍历文本串,当前遍历到的字符匹配则移动到下一个字符继续匹配;当前遍历到的字符不匹配则将模式串回退到已经匹配好的位置。
  4. 为什么根据前缀表回退有效:文本串和模式串当前遍历到的字符都匹配,然后移动到下一位,发现不匹配。那么这时就要把模式串回退到与当前文本串前一位及之前匹配好的字符串的位置
  • 根据前后缀相等得到的前缀表记录的长度保证了回退到的位置前n为字符串都是相同的。
  • 例如:当前文本串最后两位为bb,模式串为bbabb,那么根据前缀表可以将模式串回退到bba的位置,a的前面还是bb,和之前匹配失败时前面的字符串相同,那么就可以从这个位置继续之前的匹配,将a与文本串中bb后的字符进行匹配就可以了。
  • 如果继续匹配失败,那就继续往前回退,直到回退到模式串的起点。

代码

版本一

class Solution {
public://完善前缀表//next为指向前缀表中元素的指针;s为模式串  void getNext(int* next,string& s) {int j = -1;//j指向前缀的最后一个元素。j是前缀最后一个元素的下标next[0] = j;//前缀表第一个元素//从第二个元素开始完善前缀表for (int i = 1; i < s.size(); i++) {//i指向后缀的最后一个元素,也是当前遍历到字符串的位置//前后缀不相同时(j=-1时,当前字符串长度为1,前后缀必然相等)while (j >= 0 && s[i] != s[j + 1]) {j = next[j];//当遇到不匹配的情况时,查找前一位字符对应下标前缀表的元素进行回退}//前后缀相同时if (s[i] == s[j + 1]) {j++;//将j向后移动}next[i] = j;//将当前遍历到位置的字符串的最长公共字符串长度放入前缀表中}}int strStr(string haystack, string needle) {if (needle.size() == 0) return 0;vector<int> next(needle.size());//创建前缀表getNext(&next[0], needle);//将前缀表补充完整//开始匹配字符串int j = -1;for (int i = 0; i < haystack.size(); i++) {//i指向haystack//不匹配时,回退while (j >= 0 && haystack[i] != needle[j + 1]) j = next[j]; //匹配时,i和j同时向后移动if (haystack[i] == needle[j + 1]) j++;//匹配完成时,返回下标if (j == (needle.size() - 1)) return (i - needle.size() + 1);}return -1;}
};

版本二

class Solution {
public:void getNext(int* next, string& s) {int j = 0;next[0] = j;for (int i = 1; i < s.size(); i++) {//前后缀不相同while (j >= 1 && s[i] != s[j]) j = next[j - 1];//找到前一个字符对应的最长公共前后缀长度,进行回退//前后缀相同if (s[i] == s[j]) j++;next[i] = j;}}int strStr(string haystack, string needle) {if (needle.size() == 0) return 0;vector<int> next(needle.size());//前缀表getNext(&next[0], needle);//开始匹配int j = 0;for (int i = 0; i < haystack.size(); i++) {//不匹配时while (j >= 1 && haystack[i] != needle[j]) j = next[j - 1];//回退//匹配时if (haystack[i] == needle[j]) j++;//i和j同时后移//匹配完成时if (j == needle.size()) return (i - needle.size() + 1);}return -1;}
};

时间复杂度:O(n+m);匹配时只遍历一次文本串,每遍历到一个字符只做一次判断和常数级的操作。
空间复杂度:O(m);需要创建前缀表存储模式串的最长公共前后缀长度。

难点回顾

  1. 匹配过程中怎样使用前缀表:当前位置(j)字符匹配失败,找到前一个字符(j-1)在前缀表中的值,使用这个值当作索引退回到模式串的指定位置再次开始匹配。
  2. 怎么得到前缀表(实现):分两种实现。
  • 正常实现:
    前提j不光代表前缀末尾索引,还代表最长公共前后缀长度
    第一步:前缀表第一个位置初始化为0(这时只有一个字符,最长公共前后缀长度必然为0)。
    第二步:从模式串第二个位置开始(索引为1)比较前后缀末尾字符是否相同。
    第三步:当前后缀末尾元素相同时(即s[i]==s[j]),共同移动到下一个位置。同时将j(这时代表最长公共前后缀长度)赋值给前缀表索引i所在的位置。
    第四步:当前后缀末尾元素不相同时(即s[i]!=s[j]),在前缀表中查找j之前的元素对应索引位置对应的值,把这个值当作索引赋给j,完成回退操作。
    第五步:在遍历整个模式串的过程中,持续完成以上条件判断及其相应操作。
  • 减一实现:
    整体思路相同,但因为把前缀表每个数都减一,导致使用前缀表进行的回退操作略有差别:前缀表中索引位置j的值就代表要回退的位置,而不是j-1位置。
    使用j代表索引时需要加一。

这篇关于(第13、14天)【leetcode题解】#右旋字符串 28、找出字符串中第一个匹配项的下标的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java 正则表达式URL 匹配与源码全解析

《Java正则表达式URL匹配与源码全解析》在Web应用开发中,我们经常需要对URL进行格式验证,今天我们结合Java的Pattern和Matcher类,深入理解正则表达式在实际应用中... 目录1.正则表达式分解:2. 添加域名匹配 (2)3. 添加路径和查询参数匹配 (3) 4. 最终优化版本5.设计思

Java字符串操作技巧之语法、示例与应用场景分析

《Java字符串操作技巧之语法、示例与应用场景分析》在Java算法题和日常开发中,字符串处理是必备的核心技能,本文全面梳理Java中字符串的常用操作语法,结合代码示例、应用场景和避坑指南,可快速掌握字... 目录引言1. 基础操作1.1 创建字符串1.2 获取长度1.3 访问字符2. 字符串处理2.1 子字

一文详解如何在Python中从字符串中提取部分内容

《一文详解如何在Python中从字符串中提取部分内容》:本文主要介绍如何在Python中从字符串中提取部分内容的相关资料,包括使用正则表达式、Pyparsing库、AST(抽象语法树)、字符串操作... 目录前言解决方案方法一:使用正则表达式方法二:使用 Pyparsing方法三:使用 AST方法四:使用字

Java字符串处理全解析(String、StringBuilder与StringBuffer)

《Java字符串处理全解析(String、StringBuilder与StringBuffer)》:本文主要介绍Java字符串处理全解析(String、StringBuilder与StringBu... 目录Java字符串处理全解析:String、StringBuilder与StringBuffer一、St

Python中使用正则表达式精准匹配IP地址的案例

《Python中使用正则表达式精准匹配IP地址的案例》Python的正则表达式(re模块)是完成这个任务的利器,但你知道怎么写才能准确匹配各种合法的IP地址吗,今天我们就来详细探讨这个问题,感兴趣的朋... 目录为什么需要IP正则表达式?IP地址的基本结构基础正则表达式写法精确匹配0-255的数字验证IP地

MySQL更新某个字段拼接固定字符串的实现

《MySQL更新某个字段拼接固定字符串的实现》在MySQL中,我们经常需要对数据库中的某个字段进行更新操作,本文就来介绍一下MySQL更新某个字段拼接固定字符串的实现,感兴趣的可以了解一下... 目录1. 查看字段当前值2. 更新字段拼接固定字符串3. 验证更新结果mysql更新某个字段拼接固定字符串 -

Java String字符串的常用使用方法

《JavaString字符串的常用使用方法》String是JDK提供的一个类,是引用类型,并不是基本的数据类型,String用于字符串操作,在之前学习c语言的时候,对于一些字符串,会初始化字符数组表... 目录一、什么是String二、如何定义一个String1. 用双引号定义2. 通过构造函数定义三、St

golang获取当前时间、时间戳和时间字符串及它们之间的相互转换方法

《golang获取当前时间、时间戳和时间字符串及它们之间的相互转换方法》:本文主要介绍golang获取当前时间、时间戳和时间字符串及它们之间的相互转换,本文通过实例代码给大家介绍的非常详细,感兴趣... 目录1、获取当前时间2、获取当前时间戳3、获取当前时间的字符串格式4、它们之间的相互转化上篇文章给大家介

浅谈配置MMCV环境,解决报错,版本不匹配问题

《浅谈配置MMCV环境,解决报错,版本不匹配问题》:本文主要介绍浅谈配置MMCV环境,解决报错,版本不匹配问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录配置MMCV环境,解决报错,版本不匹配错误示例正确示例总结配置MMCV环境,解决报错,版本不匹配在col

详解nginx 中location和 proxy_pass的匹配规则

《详解nginx中location和proxy_pass的匹配规则》location是Nginx中用来匹配客户端请求URI的指令,决定如何处理特定路径的请求,它定义了请求的路由规则,后续的配置(如... 目录location 的作用语法示例:location /www.chinasem.cntestproxy