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

ops/2024/9/23 14:26:24/

目录

  • 右旋字符串
    • 题目描述
    • 思路
    • 代码
    • 思考
  • 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代表索引时需要加一。

http://www.ppmy.cn/ops/38592.html

相关文章

一款开源高性能AI应用框架

前言 LobeChat 是一个基于 Next.js 框架构建的 AI 会话应用&#xff0c;旨在提供一个 AI 生产力平台&#xff0c;使用户能够与 AI 进行自然语言交互。 LobeChat应用架构 LobeChat 的整体架构由前端、EdgeRuntime API、Agents 市场、插件市场和独立插件组成。这些组件相互协作&a…

封装一个可以最小化和展开的弹窗组件

gl-dialog 大概思路&#xff1a; 在弹窗组件内部引入gl-dialog-collapse&#xff0c;这个组件主要用于存储已经被最小化的弹窗&#xff08;基础数据&#xff09; 弹窗内部的数据如何在父组件拿到是通过作用域插槽来实现的 gl-dialog接收一个tempData这个数据会在内部被记录下来…

水平滑动与垂直滑动菜单

水平滑动菜单 <!DOCTYPE html> <html><head><meta charset"utf-8"><title></title><style>*{margin: 0;padding: 0;}ul{background-color: #000;}ul li{text-shadow: none;display: inline-block;height: 40px;}ul li a{…

Linux流量分析工具 | nethogs

在应急过程中&#xff0c;经常会遇到应用访问缓慢&#xff0c;网络阻塞的情况&#xff0c;分析原因可能会想到存在恶意程序把带宽占满的可能。通过这样一个小工具可以快速定位异常占用带宽程序的路径、PID、占用流量大小或是排除由带宽占满导致服务器缓慢的猜想。 一、简介 Ne…

ARM(4)缓存一致性

目录 一、缓存一致性问题 二、一致性实现方案 2.1 目录一致性协议 2.2 嗅探一致性协议 三、CHI协议 3.1 cache state 3.2 snoop维护一致性 四、其他一致性协议 4.1 MSI协议 4.2 MESI 协议 4.3 MOESI协议 本文介绍以下内容&#xff1a; 缓存一致性问题一致性实现方案…

崇贸烧录器支持PUYA普冉半导体的32位微控制器PY32F403R1DT6

芯片烧录行业领导者-崇贸技术近日发布最新的烧录软件更新及新增支持的芯片型号列表&#xff0c;其中PUYA普冉半导体的32位微控制器PY32F403R1DT6已经被崇贸的通用烧录平台AP8000所支持。 PY32F403R1DT6微控制器是基于ArmCortexM4核的32位通用微控制器产品。内置的FPU和DSP功能…

视频监控平台:交通运输标准JTT808设备SDK接入源代码函数分享

目录 一、JT/T 808标准简介 &#xff08;一&#xff09;概述 &#xff08;二&#xff09;协议特点 1、通信方式 2、鉴权机制 3、消息分类 &#xff08;三&#xff09;协议主要内容 1、位置信息 2、报警信息 3、车辆控制 4、数据转发 二、代码和解释 &#xff08;一…

计算机网络——Dijkstra路由算法

实验目的 实现基于 Dijkstra 算法的路由软件 实验内容 网络拓扑如图所示 实验过程 先编写开辟应该图的空间&#xff0c;然后给点映射数字&#xff0c;构建图。程序获取用户输入的学号&#xff0c;构建图中边的权值。接下来程序从用户输入获取最短路径的搜索起点&#xff0…