LeetCode:28. 找出字符串中第一个匹配项的下标 ——【1、理解 KMP 算法】

🍎道阻且长,行则将至。🍓

🌻算法,不如说它是一种思考方式🍀


算法专栏: 👉🏻123


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

  • 题目描述:给你两个字符串 haystackneedle ,请你在 haystack 字符串中找出 needle 字符串的第一个匹配项的下标(下标从 0 开始)。如果 needle 不是 haystack 的一部分,则返回 -1 。
  • 来源:力扣(LeetCode)
  • 难度:中等
  • 提示:
    1 <= haystack.length, needle.length <= 104
    haystackneedle 仅由小写英文字符组成

🌴解题

1.暴力法

暴力法只需要判断字符串 A 的子字符串目标字符串 needle 是不是相等即可;
i 遍历与目标字符串等长A 的子字符串haystack.substring(i,i+needle.length()) ==needle
注意到,这个遍历并不需要到字符串 A 的结尾,因为匹配长度的原因,遍历区间是 [0,haystack.length()-needle.length()]。

code:
class Solution {
    public int strStr(String haystack, String needle) {
        int ans=-1;
        for (int i = 0; i <= haystack.length()-needle.length(); i++) {
            if(haystack.substring(i,i+needle.length()).equals(needle)){
                return i;
            }
        }
        return ans;
    }
}

在这里插入图片描述

2.模式串匹配

KMP 算法

在前面使用暴力法匹配的时候是浪费了很多的匹配次数,就是指有一部分内容可以略过,不用从头开始:
在这里插入图片描述
例如上图中,遍历到第 7 个字符匹配 (b - c) 的时候,发现是不匹配的。按常规方法来说又得从字符串 A 第 2 个字符目标第一个字符开始往后面遍历,而实际上在 a b a 这一段开始的匹配是做的无用功,而有用的就是 (b - c) 前面匹配过的 a a
字符串 A 中的b前面的 a a(红框里)可以匹配上目标最前面的 a a ,字符串A的遍历就不必要回头,只有目标要返回;
例如下图:
在这里插入图片描述

即:找到目标串最前面和最后面相同的字串,这一部分不用重复,是已经验证过的。

有了前面的理解,我们就可以更好理解 KMP 算法了。
KMP 中的模式串就是目标字符串,而模式串(目标)如何回头——使用 next 数组,或说前缀表,来确定模式串的指针回到哪个位置。

前缀表就是记录一个最长公共前后缀,
前缀是指不包含最后一个字符的所有以第一个字符开头的连续子串;
后缀是指不包含第一个字符的所有以最后一个字符结尾的连续子串;
就是前缀==后缀的一个长度。

例如对于这个模式(目标)串,最长公共前后缀:
在这里插入图片描述

对于a:没有最长公共前后缀——0;
对于aa:第一个a和最后一个a——1;
对于aab:第一个a和最后一个b,不匹配——0;
对于aaba:第一个a和最后一个a——1;
对于aabaa:前面的aa和最后的aa——2,注意aab和baa、aaba和abaa不是相等的

那我们理解了这个概念,又如何用代码实现求 next 数组呢?

int[] next=new int[needle.length()];
int j=0;
next[j]=0;
for (int i = 1; i < needle.length(); i++) {
    while(j>0&&needle.charAt(j)!=needle.charAt(i)){
        j=next[j-1];
    }
    if(needle.charAt(j)==needle.charAt(i)){
        j++;
    }
    next[i]=j;
}

求 next 数组演示图:
在这里插入图片描述
根据 next 数组模式匹配:
在这里插入图片描述

code:
class Solution {
    public int strStr(String haystack, String needle) {
        int ans=-1;
        //计算前缀和next数组
        int[] next=new int[needle.length()];
        int j=0;
        next[j]=0;
        for (int i = 1; i < needle.length(); i++) {
            while(j>0&&needle.charAt(j)!=needle.charAt(i)){
                j=next[j-1];
            }
            if(needle.charAt(j)==needle.charAt(i)){
                j++;
            }
            next[i]=j;
        }

        //匹配
        j=0;
        for (int i = 0; i < haystack.length(); i++) {
            if(j< needle.length()) {
                if (haystack.charAt(i) == needle.charAt(j)) {
                    j++;
                } else if(j>0){
                    j = next[j - 1];
                    i--;
                }
            }
            if(j== needle.length()){
                ans=i-needle.length()+1;
                break;
            }
        }

        return ans;

    }
}

在这里插入图片描述


返回第一页。☝


☕物有本末,事有终始,知所先后。🍭

🍎☝☝☝☝☝我的CSDN☝☝☝☝☝☝🍓

  • 18
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 12
    评论
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Super algorithm

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值