字符串 下【KMP再续】

⚡刷题计划day9继续,可以点个免费的赞哦~

往期可看专栏,关注不迷路,

您的支持是我的最大动力🌹~

今天的重头戏是KMP算法的理解,如果是第一次接触可能会有难度,可以结合以下链接的讲解及对应视频,没有明白可以多看多听几遍,加油!然后下一期就是双指针专题了,欢迎点赞收藏关注!

(programmercarl.com)

题目一:28. 找出字符串中第一个匹配项的下标

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

(https://leetcode.cn/problems/find-the-index-of-the-first-occurrence-in-a-string/description/)

法一:一般思路

法一就是暴力解,直接遍历两个串,比较是否匹配。

详见注释

class Solution {public int strStr(String haystack, String needle) {// 获取haystack和needle的长度int n1 = haystack.length();int n2 = needle.length();
​// 如果haystack的长度小于needle的长度,直接返回-1,表示找不到if(n1 < n2){return -1;}
​// 将haystack和needle转换为字符数组char[] c1 = haystack.toCharArray();char[] c2 = needle.toCharArray();
​// 遍历haystackfor(int i = 0; i < n1; i++){// 初始化搜索的起始和结束索引int start = i;int end = i + n2 - 1;// 如果结束索引超出haystack的范围,跳出循环if(end >= n1){break;}
​// 初始化一个标志位,用于判断是否找到匹配的子串boolean flag = true;int k = 0;// 内层循环,用于比较字符while(start <= end){// 如果当前字符不匹配,设置标志位为false并跳出循环if(c1[start++] != c2[k++]){flag = false;break;}}// 如果标志位为true,表示找到了匹配的子串,返回当前索引iif(flag){return i;}}
​// 如果遍历完haystack都没有找到needle,返回-1return -1;}
}

法二:KMP算法

详见注释

class Solution {// getNext 方法用于构建 next 数组,该数组用于存储模式字符串的最长前后缀匹配的长度public void getNext(int[] next, String s) {int j = -1; // j 用于记录当前比较的子字符串的最后一个匹配位置next[0] = j; // 模式字符串的第一个字符没有前缀,所以它的 next 值是 -1for (int i = 1; i < s.length(); i++) { // 从模式字符串的第二个字符开始遍历while (j >= 0 && s.charAt(i) != s.charAt(j + 1)) { // 当前字符不匹配时,尝试回溯j = next[j]; // 根据 next 数组回溯到上一个匹配的位置}if (s.charAt(i) == s.charAt(j + 1)) { // 如果当前字符匹配,更新 jj++;}next[i] = j; // 更新 next 数组,存储当前位置的最长前后缀匹配的长度}}
​// strStr 方法用于在 haystack 字符串中查找 needle 字符串出现的位置public int strStr(String haystack, String needle) {if (haystack.length() < needle.length()) { // 如果文本字符串比模式字符串短,直接返回 -1return -1;}
​int[] next = new int[needle.length()]; // 创建 next 数组getNext(next, needle); // 构建 next 数组
​int j = -1; // 初始化 j 为 -1,表示在模式字符串中的位置for (int i = 0; i < haystack.length(); i++) { // 遍历文本字符串while (j >= 0 && haystack.charAt(i) != needle.charAt(j + 1)) { // 如果当前字符不匹配,根据 next 数组回溯j = next[j]; // 回溯到上一个匹配的位置}if (haystack.charAt(i) == needle.charAt(j + 1)) { // 如果当前字符匹配,更新 jj++;}if (j == needle.length() - 1) { // 如果 j 到达模式字符串的末尾,表示找到了匹配return (i - needle.length() + 1); // 返回模式字符串在文本字符串中的起始位置}}
​return -1; // 如果遍历完文本字符串都没有找到匹配,返回 -1}
}

题目二:459.重复的子字符串

459.重复的子字符串

(https://leetcode.cn/problems/repeated-substring-pattern/description/)

法一:暴力解法

解题思路:

如果字符串 S 包含一个重复的子字符串,则可以通过多次移位原字符串,最终能得到一新相同字符串与原字符串匹配。

例:abcabc

移位一次:cabcab 移位两次:bcabca 移位三次:abcabc

现在字符串和原字符串匹配了,所以可以得出结论存在重复的子串。

但是对于重复字符串比较长的情况,效率就会很低,并且超时。

基于此思想,可以新建一个字符串str = s + s,等于原字符串s再加上自身,这样新的字符串str就包含了所有移动的情况。

比如字符串:s = acd,那么 str = s + s = acdacd

acd 移动的可能:dac、cda。其实都包含在了 str 中了。就像一个滑动窗口

一开始 acd (acd) ,移动一次 ac(dac)d,移动两次 a(cda)cd。循环结束.

注意需要去除 str 中首尾元素

 所以可以直接判断之后,是否包含自身元素。如果包含。则表明存在重复子串。

附一网友理解:如果S不包含重复的子字符串,则S本身就是所谓的“重复子字符串”(这里方便自己理解,视为重复一次,不深究= =),str = S+S,说明S是str的重复子字符串,刨去str首尾两个字符之后(相当于分别破坏前一个S头部和后一个S尾部),不能满足str包含S。

AC代码

class Solution {public boolean repeatedSubstringPattern(String s) {String t = s + s;t = t.substring(1, t.length() - 1); 	// 掐头去尾if (t.contains(s)) {return true;}return false;}
}

法二:KMP

关于kmp还不是很理解的可以再康康上期附送的链接,这里不再多叙述

然后此题需要证明,有兴趣的小伙伴也可以康康这个链接:

(https://programmercarl.com/0459.重复的子字符串.html#思路)

我们这里直接给出判断的结论:

如果len % (len - (next[len - 1] + 1)) == 0 ,即数组的长度正好可以被 最长相等前后缀不包含的子串的长度 整除 ,说明该字符串有重复的子字符串。

class Solution {public void getNext(int[] next,String s){int j=-1;next[0]=j;for(int i=1;i<s.length();i++){while (j>=0 && s.charAt(i)!=s.charAt(j+1)){j = next[j];}if(s.charAt(i)==s.charAt(j+1)){j++;}next[i] = j;}}public boolean repeatedSubstringPattern(String s) {if(s.equals("")) return false;int len = s.length();int j=-1;int[] next = new int[len];getNext(next,s);////1.如果 next[len - 1] != -1,则说明字符串有最长相同的前后缀//2.最长相等前后缀的长度为:next[len - 1] + 1。(这里的next数组是以统一减一的方式计算的,因此需要+1//3.len - (next[len - 1] + 1) 是最长相等前后缀不包含的子串的长度。if(next[len-1] != -1 && len % (len-(next[len-1]+1))==0){return true;}return false;}
}

下期开启新专题,感谢点赞收藏关注哦🌹

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

LeetCode 面试经典150题 190.颠倒二进制位

复习知识&#xff1a;正数的原码、反码、补码相同&#xff0c;负数的反码在其原码的基础上, 符号位不变&#xff0c;其余各个位取反&#xff0c;负数的补码是在其原码的基础上, 符号位不变, 其余各位取反, 最后1 (即在反码的基础上1)。 题目&#xff1a;颠倒给定的 32 位无符号…

【SQLite数据库常规使用命令】

之前在做围绕数据库相关的一些小工具的时候&#xff0c;想找一款数据库作为小工具的资料库。需求是&#xff1a;不用复杂的安装&#xff0c;支持简单SQL&#xff0c;空间占用小&#xff0c;操作简单等等。 结合着之前接触到的一些研发同事做的产品的使用经验&#xff0c;我想到…

华为HarmonyOS地图服务 3 - 如何开启和展示“我的位置”?

一. 场景介绍 本章节将向您介绍如何开启和展示“我的位置”功能&#xff0c;“我的位置”指的是进入地图后点击“我的位置”显示当前位置点的功能。效果如下&#xff1a; 二. 接口说明 “我的位置”功能主要由MapComponentController的方法实现&#xff0c;更多接口及使用方法…

基于LSTM的温度时序预测

1.背景 本文接【时序预测SARIMAX模型】 一文&#xff0c;采用LSTM模型进行平均温度数据预测。具体的背景和数据分析就不做重复说明&#xff0c;感兴趣可以去看上文即可。 2.LSTM模型 RNN&#xff08;Recurrent Neural Network&#xff0c;循环神经网络&#xff09;是一种特殊…

【ARM】armv8的虚拟化深度解读

Type-1 hypervisor Type-1虚拟化也叫做Bare metal, standalone, Type1 Type2 hypervisor Type-2虚拟化也叫做hosted, Type-2 VM和vCPU(虚拟机和虚拟cpu) 在一个VM&#xff08;虚拟机&#xff09;中有多个vCPU&#xff0c;多个vCPU可能属于同一个Vritual Processor。 EL2…

java-----异常

目录 异常&#xff1a;代表程序出现的问题 运行时异常和编译时异常的区别&#xff1f; 异常的作用&#xff1a; 异常的处理方式: 异常中常见的方法: 抛出异常: 自定义异常: 异常&#xff1a;代表程序出现的问题 Exception:叫做异常&#xff0c;代表程序可能出现的问题。…

【alluxio编译报错】Some files do not have the expected license header

Some files do not have the expected license header 快捷导航 在开始解决问题之前&#xff0c;大家可以通过下面的导航快速找到相关资源啦&#xff01;&#x1f4a1;&#x1f447; 快捷导航链接地址备注相关文档-ambaribigtop自定义组件集成https://blog.csdn.net/TTBIGDA…

【JavaScript】LeetCode:46-50

文章目录 46 翻转二叉树47 对称二叉树48 二叉树的直径49 二叉树的层序遍历50 将有序数组转换为二叉搜索树 46 翻转二叉树 递归前序遍历 / 后序遍历&#xff0c;这里给出前序遍历的代码。遍历节点&#xff0c;交换左右子树。 /*** Definition for a binary tree node.* functio…

vue3快速入门(看心情更新)

vue3初始化工程目录 编写一个App .vscode下的extensions.json 配置插件的地方 public 页签图标 src 你的.vue文件都是在这个目录下的 .gitgnore 忽略文件 env.d.ts 让Ts去识别一些文件 index.html 入口文件 vite.config.ts 整个工程的配置文件 .vue文件中可以写的内容 template…

Windows安装Oracle11gR2(图文教程)

本章教程&#xff0c;记录在Windows10上安装Oracle11gR2过程。 一、下载安装包 通过网盘分享的文件&#xff1a;oracle11g 链接: https://pan.baidu.com/s/15ilciQ5NlKWtClklmdAH_w?pwds4dd 提取码: s4dd 二、下载并解压文件 将网盘中的安装包文件下载到本地&#xff0c;在此之…

心觉:感恩何其重要,感恩之心如何培养

Hi&#xff0c;我是心觉&#xff0c;与你一起玩转潜意识、脑波音乐和吸引力法则&#xff0c;轻松掌控自己的人生&#xff01; 挑战每日一省写作177/1000天 上篇文章我们讲了保持感恩之心&#xff0c;可以吸引更多的机会和财富 但是现实中很多人是缺乏感恩之心 这是由于他们…

c++day3 手动封装一个顺序表(SeqList),分文件编译实现

要求: 有私有成员&#xff1a;顺序表数组的起始地址 ptr、 顺序表的总长度&#xff1a;size、顺序表的实际长度&#xff1a;len 成员函数&#xff1a;初始化 init(int n) 判空&#xff1a;empty 判满&#xff1a;full 尾插&#xff1a;push_back 插入&#xff1a;insert&…

进程间的通信4 共享内存

共享内存 1.共享内存简介 共享内存是将分配的物理空间直接映射到进程的用户虚拟地址空间中&#xff0c;减少数据在内核空间缓存共享内存是一种效率较高的进程间通讯的方式在 Linux 系统中通过 ipcs -m 查看所有的共享内存 共享内存模型图 2.共享内存的创建 1.函数头文件 #…

【如何在 Windows 10 主机上通过 VMware 安装 Windows 11 虚拟机,并共享主机网络】

环境说明 主机操作系统&#xff1a;Windows 10虚拟机操作系统&#xff1a;Windows 11虚拟机软件&#xff1a;VMware 步骤一&#xff1a;确保主机&#xff08;Windows 10&#xff09;网络连接正常 启动网络加速软件&#xff1a;在主机上启动软件&#xff0c;确保主机可以正常访…

JavaEE: 深入探索TCP网络编程的奇妙世界(三)

文章目录 TCP核心机制TCP核心机制三: 连接管理建立连接(三次握手)断开连接(四次挥手)三次握手/四次挥手 流程简图 TCP核心机制 书接上文~ TCP核心机制三: 连接管理 建立连接(三次握手),断开连接(四次挥手). 这里的次数指的是网络通信的次数,挥手/握手是形象的比喻(handshake…

基于SpringBoot+Vue的智慧物业管理系统

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、SSM项目源码 精品专栏&#xff1a;Java精选实战项目源码、Python精…

SpringBoot3核心特性-核心原理

目录 传送门前言一、事件和监听器1、生命周期监听2、事件触发时机 二、自动配置原理1、入门理解1.1、自动配置流程1.2、SPI机制1.3、功能开关 2、进阶理解2.1、 SpringBootApplication2.2、 完整启动加载流程 三、自定义starter1、业务代码2、基本抽取3、使用EnableXxx机制4、完…

SaaS软件的配置化平台是如何实现个性化定制的?

SaaS&#xff08;Software as a Service&#xff0c;软件即服务&#xff09;是一种通过互联网提供软件的模式&#xff0c;用户无需安装和维护任何复杂的基础设施&#xff0c;只需通过网络连接即可使用软件。SaaS 供应商负责软件的维护、升级和可用性&#xff0c;用户则通过订阅…

智能体时代,AI正从“神坛”走向“人间”

从通用大模型到行业大模型&#xff1a;AI智能体引领新风口 在人工智能领域&#xff0c;一场深刻的变革正悄然发生。从昔日高高在上的通用大模型&#xff0c;到如今愈发接地气的行业大模型&#xff0c;AI的风向标已经鲜明地指向了AI智能体&#xff08;AI Agent&#xff09;&…

APO v0.4.0 发布:新增影响面分析;新增调用数据库指标;优化告警事件关联展示

APO 新版本 v0.4.0 正式发布&#xff01;本次更新主要包含以下内容&#xff1a; 新增影响面分析&#xff0c;识别服务端点对服务入口的影响 服务入口是指业务被访问时调用的第一个服务端点&#xff0c;在调用拓扑图中处于最上游。服务入口直接反映了系统对外提供服务的状态&a…