(滑动窗口) 76. 最小覆盖子串 ——【Leetcode每日一题】

这篇具有很好参考价值的文章主要介绍了(滑动窗口) 76. 最小覆盖子串 ——【Leetcode每日一题】。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

❓76. 最小覆盖子串

难度:困难

给你一个字符串 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 < = 1 0 5 1 <= m, n <= 10^5 1<=m,n<=105
  • st 由英文字母组成

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

💡思路:滑动窗口

滑动窗口的思想
lr 表示滑动窗口的左边界右边界,通过改变 l, r扩展收缩 滑动窗口,可以想象成一个窗口在字符串上游走,当这个窗口包含的元素满足条件,即包含字符串 t 的所有元素,记录下这个滑动窗口的长度 r - l +1,这些长度中的最小值就是要求的结果。

由于 t 可能包含 重复字符 ,所以使用哈希表 ori 记录 t 中所有的字符以及它们的个数;使用另一个哈希表 cnt 动态维护窗口中所有的字符以及它们的个数。

如果 当前窗口 中包含 t 中所有的字符,且对应的字符个数都不小于 t 的哈希表中各个字符的个数,则当前窗口是「可行」的。此时左边界 l 右移,直到不涵盖 t 中的所有字符,上一个即为满足条件的 当前窗口的最小子串

使用 distance 记录 当前窗口 中字符 属于 t 对应的个数;当 cnt[s[r]] < ori[s[r]] 时,每增加一个 s[r]distance ++ ,否则 distance 不变。 同理 当 cnt[s[l]] <= ori[s[l]] 每减少一个 s[l]distance -- ,否则 distance 不变。(这样可以在 O ( 1 ) O(1) O(1) 的复杂度判断 当前窗口 内的字符是否符合要求

具体步骤如下

  1. 不断增加 r 使滑动窗口 增大,直到窗口包含了 t 的所有元素;
  2. 不断增加 l 使滑动窗口 缩小,因为是要求最小字串,所以将不必要的元素排除在外,使长度减小,直到碰到一个必须包含的元素,这个时候不能再扔了,再扔就不满足条件了,记录此时滑动窗口的长度,并保存最小值;
  3. l 再增加一个位置,这个时候滑动窗口肯定不满足条件了,那么继续从 步骤 1 开始执行,寻找新的满足条件的滑动窗口,如此反复,直到 r 超出了字符串 s 范围。

🍁代码:(C++、Java)

C++

class Solution {
public:
    unordered_map <char, int> ori, cnt;

    string minWindow(string s, string t) {
        for(const auto &c : t){ //使用哈希表记录t中所有的字符,及其对应的个数
            ++ori[c];
        }
        int l = 0, r = -1;
        int len = INT_MAX, ansl = -1;
        int distance = 0;

        while(r < int(s.size())){
            if(ori.find(s[++r]) != ori.end() && ++cnt[s[r]] <= ori[s[r]] ){//对任何属于t的字符都记录
                distance++;//记录当前窗口的子串中属于t的字符的个数
            }
            while(distance == t.size() && l <= r){//包括所有字符,左指针右移
                if(r - l + 1 < len){//更新结果
                    len = r - l + 1;
                    ansl = l;
                }
                if(ori.find(s[l]) != ori.end() && --cnt[s[l]] < ori[s[l]]){
                    distance--;
                }
                ++l;
            }
        }
        return ansl == -1 ? string() : s.substr(ansl, len);
    }
};

Java

class Solution {
    Map<Character, Integer> ori = new HashMap<Character, Integer>();
    Map<Character, Integer> cnt = new HashMap<Character, Integer>();
    public String minWindow(String s, String t) {
        for(int i = 0; i < t.length(); i++){//使用哈希表记录t中所有的字符,及其对应的个数
            char c = t.charAt(i);
            ori.put(c, ori.getOrDefault(c, 0) + 1);
        }
        
        int l = 0, r = -1;
        int len = Integer.MAX_VALUE, ansl = -1;
        int distance = 0;
        while(++r < s.length()){
            char c = s.charAt(r);
            if(ori.containsKey(c)){//对任何属于t的字符都记录
                cnt.put(c, cnt.getOrDefault(c, 0) + 1);
                if(cnt.get(c) <= ori.get(c)){//记录当前窗口的子串中属于t的字符的个数
                    distance++;
                }
            }
            while(distance == t.length() && l <= r){
                if(r - l + 1 < len){//更新结果
                    len = r - l + 1;
                    ansl = l;
                }
                if(ori.containsKey(s.charAt(l))){
                    cnt.put(s.charAt(l), cnt.getOrDefault(s.charAt(l), 0) - 1);
                    if(cnt.get(s.charAt(l)) < ori.get(s.charAt(l))){
                        distance--;
                    }
                }
                ++l;
            }
        }
        return ansl == -1 ? "" : s.substring(ansl, ansl + len);
    }
}
🚀 运行结果:

(滑动窗口) 76. 最小覆盖子串 ——【Leetcode每日一题】,LeetCode,leetcode,算法,职场和发展

🕔 复杂度分析:
  • 时间复杂度 O ( ∣ s ∣ + ∣ t ∣ ) O(∣s∣+∣t∣) O(s+t),最坏情况下左右指针对 s 的每个元素各遍历一遍,哈希表中对 s 中的每个元素各插入、删除一次,对 t 中的元素各插入一次。
  • 空间复杂度 O ( C ) O(C) O(C),这里用了两张哈希表作为辅助空间,每张哈希表最多不会存放超过字符集大小的键值对,我们设字符集大小为 C ,则渐进空间复杂度为 O ( C ) O(C) O(C)

题目来源:力扣。

放弃一件事很容易,每天能坚持一件事一定很酷,一起每日一题吧!
关注我LeetCode主页 / CSDN—力扣专栏,每日更新!文章来源地址https://www.toymoban.com/news/detail-718360.html

注: 如有不足,欢迎指正!

到了这里,关于(滑动窗口) 76. 最小覆盖子串 ——【Leetcode每日一题】的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • LeetCode----76. 最小覆盖子串

     题目 给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串,则返回空字符串 “” 。 注意: 示例 1: 输入:s = “ADOBECODEBANC”, t = “ABC” 输出:“BANC” 解释:最小覆盖子串 “BANC” 包含来自字符串 t 的 ‘A’、

    2024年02月06日
    浏览(44)
  • leetcode做题笔记76最小覆盖子串

    给你一个字符串  s  、一个字符串  t  。返回  s  中涵盖  t  所有字符的最小子串。如果  s  中不存在涵盖  t  所有字符的子串,则返回空字符串  \\\"\\\"  。 注意: 对于  t  中重复字符,我们寻找的子字符串中该字符数量必须不少于  t  中该字符数量。 如果  s  中存在

    2024年02月13日
    浏览(40)
  • 算法leetcode|76. 最小覆盖子串(rust重拳出击)

    给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串,则返回空字符串 \\\"\\\" 。 注意 : 对于 t 中重复字符,我们寻找的子字符串中该字符数量必须不少于 t 中该字符数量。 如果 s 中存在这样的子串,我们保证它是唯

    2024年02月10日
    浏览(61)
  • 【leetcode题解C++】51.N皇后 and 76.最小覆盖子串

    51. N皇后 按照国际象棋的规则,皇后可以攻击与之处在同一行或同一列或同一斜线上的棋子。 n 皇后问题  研究的是如何将  n  个皇后放置在  n×n  的棋盘上,并且使皇后彼此之间不能相互攻击。 给你一个整数  n  ,返回所有不同的  n   皇后问题  的解决方案。 每一种

    2024年02月20日
    浏览(41)
  • LeetCode热题HOT100:76. 最小覆盖子串,84.柱状图中最大的矩形、96. 不同的二叉搜索树

    题目 :给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串,则返回空字符串 “” 。 注意: 对于 t 中重复字符,我们寻找的子字符串中该字符数量必须不少于 t 中该字符数量。 如果 s 中存在这样的子串,我们保

    2023年04月19日
    浏览(56)
  • 2023-07-13 LeetCode每日一题(下降路径最小和)

    点击跳转到题目位置 给你一个 n x n 的 方形 整数数组 matrix ,请你找出并返回通过 matrix 的 下降路径 的 最小和 。 下降路径 可以从第一行中的任何元素开始,并从每一行中选择一个元素。在下一行选择的元素和当前行所选元素最多相隔一列(即位于正下方或者沿对角线向左

    2024年02月15日
    浏览(78)
  • 【Leetcode刷题-Python/C++】长度最小的子数组(滑动窗口)

    209.长度最小的子数组 给定一个含有 n 个正整数的数组和一个正整数 target 。 找出该数组中满足其和 ≥ target 的长度最小的 连续子数组 [numsl, numsl+1, …, numsr-1, numsr] ,并返回其长度。如果不存在符合条件的子数组,返回 0 。 输入:target = 7, nums = [2,3,1,2,4,3] 输出:2 解释:子数

    2023年04月08日
    浏览(41)
  • 【算法|滑动窗口No.1】leetcode209. 长度最小的子数组

    个人主页:兜里有颗棉花糖 欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 兜里有颗棉花糖 原创 收录于专栏【手撕算法系列专栏】【LeetCode】 🍔本专栏旨在提高自己算法能力的同时,记录一下自己的学习过程,希望对大家有所帮助 🍓希望我们一起努力、成长,共同进步。

    2024年02月08日
    浏览(41)
  • 每日一题——LeetCode1266.访问所有点的最小时间

    方法一 个人方法 找规律: 当前的点为current,下一个点为next,x为两点横坐标之间距离,y为两点竖坐标之间距离 1、当两点横坐标相同时,两点距离为y 2、当两点竖坐标相同时,两点距离为x 3、当两点x与y相同时,则两点的连线在直线y=x上,那么两点的距离既是x也是y 4、当

    2024年01月20日
    浏览(51)
  • ( 数组) 209. 长度最小的子数组——【Leetcode每日一题】

    难度:中等 给定一个含有 n 个正整数的数组和一个正整数 target 。 找出该数组中满足其和 ≥ target 的长度最小的 连续子数组 [numsl, numsl+1, ..., numsr-1, numsr] ,并返回其长度。如果不存在符合条件的子数组,返回 0 。 示例 1: 输入:target = 7, nums = [2,3,1,2,4,3] 输出:2 解释:子数

    2024年02月06日
    浏览(39)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包