Leetcode刷题详解——最小覆盖子串

1. 题目链接:76. 最小覆盖子串

2. 题目描述:

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

注意:

  • 对于 t 中重复字符,我们寻找的子字符串中该字符数量必须不少于 t 中该字符数量。
  • 如果 s 中存在这样的子串,我们保证它是唯一的答案。

示例 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 的子串中,
因此没有符合条件的子字符串,返回空字符串。

提示:

  • m == s.length
  • n == t.length
  • 1 <= m, n <= 105
  • st 由英文字母组成

3. 解法(滑动窗口+哈希表)

3.1 算法思路:

  1. 创建两个哈希表,其中一个将目标串的信息统计起来,另一个哈希表动态的维护窗口内字符串的信息
  2. 当动态哈希表中包含目标串中所有的字符,并且对应的个数都不小于目标串的哈希表中各字符的个数,那么当前的窗口就是一种可行的方案

3.2 算法流程:

  1. 定义两个辅助数组:hash1hash2hash1用于统计字符串t中每个字符的频次,hash2用于统计窗口内每个字符的频次。

  2. 使用变量kinds统计字符串t中有效字符的种类数。遍历字符串t,若hash1[ch]++ == 0,则表示该字符是新字符,将kinds加1。

  3. 初始化变量minlen为最大整数,表示最小窗口的长度,begin为-1,表示最小窗口的起始位置。

  4. 使用双指针leftright,遍历字符串s。对于每个字符in=s[right],将其在hash2中的频次加1。

  5. 如果hash2[in] == hash1[in],说明窗口内的某个字符in已经达到了所需的频次,将count加1。

  6. 如果count等于kinds,说明窗口内包含了字符串t中的所有有效字符。此时需要更新结果。

  7. 如果当前窗口的长度小于minlen,将minlen更新为当前窗口的长度,begin更新为当前窗口的起始位置。

  8. 将窗口左边界向右移动,即将字符s[left++]移出窗口。如果hash2[out] == hash1[out] - 1,说明窗口内的字符out频次减少后仍然满足要求,将count减1。

  9. 重复步骤4至8,直到right遍历完整个字符串s

  10. 最后判断begin是否仍然为-1,如果是则表示未找到满足要求的最小窗口子串,返回空字符串。否则,返回字符串s中起始位置为begin,长度为minlen的子串作为结果。

3.3 C++算法代码:

c++ 复制代码
class Solution {
public:
    string minWindow(string s, string t) {
        int hash1[128]={0};//统计字符串t中每个字符的频次
        int kinds=0;//统计有效字符有多少种
        for(auto ch:t)
            if(hash1[ch]++==0) kinds++;
        int hash2[128]={0};//统计窗口内每个字符的频次
        int minlen=INT_MAX,begin=-1;
        for(int left=0,right=0,count=0;right<s.size();right++)
        {
            char in=s[right];
            if(++hash2[in]==hash1[in]) count++;//进窗口+维护count
            while(count==kinds) //判断条件
            {
                 if(right-left+1<minlen)//更新结果
                 {
                     minlen=right-left+1;
                     begin=left;
                 }
                 char out=s[left++];
                 if(hash2[out]--==hash1[out]) count--;
            }
        }
        if(begin==-1) return "";
        else return s.substr(begin,minlen);
    }
};
相关推荐
夏鹏今天学习了吗5 小时前
【LeetCode热题100(82/100)】单词拆分
算法·leetcode·职场和发展
mit6.8246 小时前
mysql exe
算法
2501_901147836 小时前
动态规划在整除子集问题中的应用与高性能实现分析
算法·职场和发展·动态规划
中草药z7 小时前
【嵌入模型】概念、应用与两大 AI 开源社区(Hugging Face / 魔塔)
人工智能·算法·机器学习·数据集·向量·嵌入模型
CCPC不拿奖不改名7 小时前
SQL基础(SQL小白教程):MySQL语句+环境一键搭建+面试习题
数据库·sql·计算机网络·mysql·oracle·面试·职场和发展
踩坑记录7 小时前
leetcode hot100 189.轮转数组 medium
leetcode
知乎的哥廷根数学学派7 小时前
基于数据驱动的自适应正交小波基优化算法(Python)
开发语言·网络·人工智能·pytorch·python·深度学习·算法
ADI_OP7 小时前
ADAU1452的开发教程10:逻辑算法模块
算法·adi dsp中文资料·adi dsp·adi音频dsp·adi dsp开发教程·sigmadsp的开发详解
xingzhemengyou18 小时前
C语言 查找一个字符在字符串中第i次出现的位置
c语言·算法
Dream it possible!8 小时前
LeetCode 面试经典 150_二分查找_在排序数组中查找元素的第一个和最后一个位置(115_34_C++_中等)
c++·leetcode·面试