LeetCode No28. 实现 strStr()

题目

实现 strStr() 函数。

给你两个字符串 haystack 和 needle ,请你在 haystack 字符串中找出 needle 字符串出现的第一个位置(下标从 0 开始)。如果不存在,则返回  -1 。

说明:

当 needle 是空字符串时,我们应当返回什么值呢?这是一个在面试中很好的问题。

对于本题而言,当 needle 是空字符串时我们应当返回 0 。这与 C 语言的 strstr() 以及 Java 的 indexOf() 定义相符。

示例 1:

输入:haystack = "hello", needle = "ll"
输出:2

示例 2:

输入:haystack = "aaaaa", needle = "bba"
输出:-1

示例 3:

输入:haystack = "", needle = ""
输出:0

提示:

1 <= haystack.length, needle.length <= 10^4
haystack 和 needle 仅由小写英文字符组成

思路

暴力

简单暴力,可以遍历每个下标,然后截取接下来的n位,看字符子串和给的匹配串是否匹配即可,我这里偷了下懒,用substring方法,应该是要遍历构建新的字符的,哈哈。

KMP

这个题目其实就是经典的KMP入门题目,可惜我之前对这个算法已经忘记了差不多了,这次也是看大佬的代码再温习了一遍了。

AC代码

暴力

点击查看代码
class Solution {
    public int strStr(String haystack, String needle) {
        int n = needle.length();
        if( n==0 || haystack.equals(needle) ) {
            return 0;
        }
        for(int i=0; i+n<=haystack.length(); i++) {
            if( needle.equals(haystack.substring(i, i+n)) ) {
                return i;
            }
        }
        return -1;
    }
}

KMP

点击查看代码
class Solution {
    // KMP 算法
    // ss: 原串(string)  pp: 匹配串(pattern)
    public int strStr(String ss, String pp) {
        if (pp.isEmpty()) return 0;
        
        // 分别读取原串和匹配串的长度
        int n = ss.length(), m = pp.length();
        // 原串和匹配串前面都加空格,使其下标从 1 开始
        ss = " " + ss;
        pp = " " + pp;

        char[] s = ss.toCharArray();
        char[] p = pp.toCharArray();

        // 构建 next 数组,数组长度为匹配串的长度(next 数组是和匹配串相关的)
        int[] next = new int[m + 1];
        // 构造过程 i = 2,j = 0 开始,i 小于等于匹配串长度 【构造 i 从 2 开始】
        for (int i = 2, j = 0; i <= m; i++) {
            // 匹配不成功的话,j = next(j)
            while (j > 0 && p[i] != p[j + 1]) j = next[j];
            // 匹配成功的话,先让 j++
            if (p[i] == p[j + 1]) j++;
            // 更新 next[i],结束本次循环,i++
            next[i] = j;
        }

        // 匹配过程,i = 1,j = 0 开始,i 小于等于原串长度 【匹配 i 从 1 开始】
        for (int i = 1, j = 0; i <= n; i++) {
            // 匹配不成功 j = next(j)
            while (j > 0 && s[i] != p[j + 1]) j = next[j];
            // 匹配成功的话,先让 j++,结束本次循环后 i++
            if (s[i] == p[j + 1]) j++;
            // 整一段匹配成功,直接返回下标
            if (j == m) return i - m;
        }

        return -1;
    }
}
posted @ 2022-04-26 22:21  Asimple  阅读(22)  评论(0编辑  收藏  举报