【LC100】No76. 最小覆盖子串

题目描述

给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串,则返回空字符串 ""


注意:

  • 对于 t 中重复字符,我们寻找的子字符串中该字符数量必须不少于 t 中该字符数量。

  • 如果 s 中存在这样的子串,我们保证它是唯一的答案。

 

提示:

  • m == s.length

  • n == t.length

  • 1 <= m, n <= 105

  • st 由英文字母组成

进阶:

你能设计一个在 o(m+n) 时间内解决此问题的算法吗?

示例

示例 1:

输入:s = "ADOBECODEBANC", t = "ABC"
输出:"BANC"
解释:最小覆盖子串 "BANC" 包含来自字符串 t 的 'A'、'B' 和 'C'。

示例 2:

输入:s = "a", t = "a"
输出:"a"
解释:整个字符串 s 是最小覆盖子串。

示例 3:

输入: s = "a", t = "aa"
输出: ""
解释: t 中两个字符 'a' 均应包含在 s 的子串中,
因此没有符合条件的子字符串,返回空字符串。

链接

https://leetcode.cn/problems/minimum-window-substring/description/?envType=study-plan-v2&envId=top-100-liked

思路

做出来了吗?如做。

又见子串问题,经过前几天的锤炼,首先想到滑动窗口。

但本题的特殊点在于:s 的子串只需要覆盖 t 即可,也就是说 s 的子串可能大于 t 的长度,但只要全覆盖即可。如前文示例一。

因此,我们需要转换处理思路,不能通过判断滑动窗口每次进入的元素是否在 t 中来移除窗口左端点。

我们可以换一个角度处理:s 的子串只需要包含 t 中所有字母且出现次数一致

解法一:变长滑动窗口

依旧是枚举滑动窗口的右端点,当窗口覆盖 t,则从左端点缩小窗口,直到不覆盖为止。

为了便于判断是否覆盖,我们将 s 和 t 分别用数组保存每个字母的出现次数,只要判断数组是否一致即可。

new int[128] 是因为 128 包含了ASCII所有字符,便于处理。

设置两个下标 resLeft 和 resRight 分别存储最短子串的左右下标,其中设置 resLeft = -1 是为了便于最终的判断。当循环结束后,resLeft 仍然 < 0,代表在 s 中没有找到可以覆盖 t 的子串,返回 0。

代码

class Solution {
    public String minWindow(String s, String t) {
        if (s == null || t == null || s.length() < t.length()) {
            return "";
        }
        // 最终子串的左下标和右下标
        int resLeft = -1, resRight = s.length();
        int[] cnt = new int[128];
        int[] cns = new int[128];
        // 统计t中各字母的出现次数
        char[] tArr = t.toCharArray();
        for (char c : tArr) {
            cnt[c]++;
        }
        char[] sArr = s.toCharArray();
        int left = 0;
        // 滑动窗口枚举右端点
        for (int right = 0; right < s.length(); right++) {
            cns[sArr[right]]++;
            while (isCovered(cns, cnt)) { 
                // 只要当前窗口覆盖了t,就判断当前长度是否是最短
                if (right - left < resRight - resLeft) {
                    resLeft = left;
                    resRight = right;
                }
                // 左端点收缩
                cns[sArr[left]]--;
                left++;
            }
        }
        // 小于0代表没有结果
        return resLeft < 0 ? "" : s.substring(resLeft, resRight + 1);
    }

    private boolean isCovered(int[] cns, int[] cnt) {
        for (int i = 0; i < 128; i++) {
            if (cns[i] < cnt[i]) {
                return false;
            }
        }
        return true;
    }
}
  • 时间复杂度:O(128m + n),left 每增加一次,我们需要 O(128) 的时间判断窗口是否覆盖

  • 空间复杂度:O(128)

解法二:变长滑动窗口优化

解法一中,在循环的内部循环中我们套了一个对两个字母出现频次数组的循环。

如何进一步优化这点,也就是考虑“如何做才能每次无需对字母所有出现频次循环?”。

题目需要的是“s 的子串,包含 t 中所有字母,且出现次数一致”,这其中字母的出现次数我们已经能通过 cnt 数组的加减来处理判断。那么我们只需要再找到一个方法,用来判断“s 的子串覆盖了 t 中字母的种类”即可。

同样的加减思路,我们使用一个变量 count 来统计 t 中出现的字母种类数。当 s 的滑动窗口移动时,只要每次都使得 cnt 数组元素有加减变动,就对种类数 count 进行一次判断。

cnt 数组表示 t 中字母的出现次数,在初始化 cnt 数组时,每出现一种新字母,我们令 count++。在 s 窗口滑动时,窗口中每出现一个字母的出现次数符合了 t 中的出现次数,就令 count--。

最后当种类数 count 为 0 时,即代表窗口全覆盖了 t。

代码

class Solution {
    public String minWindow(String s, String t) {
        if (s == null || t == null || s.length() < t.length()) {
            return "";
        }
        // 最终子串的左下标和右下标
        int resLeft = -1, resRight = s.length();
        // 统计t中出现的字母种类数
        int count = 0;
        // 统计t中各字母的出现次数
        int[] cnt = new int[128];
        char[] tArr = t.toCharArray();
        for (char c : tArr) {
            if (cnt[c] == 0) {
                count++;
            }
            cnt[c]++;
        }
        char[] sArr = s.toCharArray();
        int left = 0;
        // 每次窗口滑动导致cnt数组变动,都需要对种类数判断
        for (int right = 0; right < s.length(); right++) {
            cnt[sArr[right]]--;
            if (cnt[sArr[right]] == 0) {
                count--;
            }
            // s的子串符合了t
            while (count == 0) {
                if (right - left < resRight - resLeft) {
                    resLeft = left;
                    resRight = right;
                }
                if (cnt[sArr[left]] == 0) {
                    count++;
                }
                cnt[sArr[left]]++;
                left++;
            }

        }
        return resLeft < 0 ? "" : s.substring(resLeft, resRight + 1);
    }
}
  • 时间复杂度:O(m + n),m 为 s 的长度,n 为 t 的长度

  • 空间复杂度:O(128)


【LC100】No76. 最小覆盖子串
https://tiamo495.com//archives/lc100-no.76.-zui-xiao-fu-gai-zi-chuan
作者
tiamo495
发布于
2025年07月24日
许可协议