leetcode 76. 最小覆盖子串(滑动窗口)

leetcode 76. 最小覆盖子串(滑动窗口)

给你一个字符串 S、一个字符串 T,请在字符串 S 里面找出:包含 T 所有字符的最小子串。
示例:

输入: S = “ADOBECODEBANC”, T = “ABC”
输出: “BANC”
说明:

如果 S 中不存这样的子串,则返回空字符串 “”。
如果 S 中存在这样的子串,我们保证它是唯一的答案。

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/minimum-window-substring
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

  1. 暴力解法
    将字符串S划分成长度大于等于T的子串,一个个判断是否符合条件。
    可以注意到,找到的字符串并没有严格按照字符串T的顺序,我们在判断是否符合条件的话就需要分别统计字符串中每个字符出现的次数,建立字符频数数组,然后逐个比较。
    此方法做了很多没必要的比较,费时费力

  2. 滑动窗口
    这个题目是典型的滑动窗口题。
    我们先将窗口逐步覆盖S中第一个符合条件的字串,如图
    first
    然后再将左边界逐步右移至刚好使窗口符合条件的位置,此时为当前的最小字串,如图
    second
    为了寻找后面的最小字串,再将左边界右移一步,此时窗口不符合条件,如图
    third
    再将窗口右边界右移,直至符合条件,如图
    forth
    继续将左边界右移直至刚好符合条件,比较此时的长度与之前找到的字串长度,取最小值
    fifth
    继续将左边界右移一步,此时窗口不符合条件,再将右边界右移直至符合条件,再对左边进行调整,最后的结果便是题目的输出
    end
    代码如下:

public String minWindow(String s, String t) { 
        int sLen = s.length();
        int tLen = t.length();
        if (sLen == 0 || tLen == 0 || sLen < tLen) {
            return "";
        }
        char[] ss = s.toCharArray();//输入字符串
        char[] tt = t.toCharArray();//目标字符串
        int[] win = new int[128];//s的滑动窗口的字符频度记录表,字母z的ASCII为122,我们直接使用字母的ASCII作为下标
        int[] tAll = new int[128];//目标字符串的字符频度记录表
        for(char c:tt)
            tAll[c]++;//直接将字符c转化为ASCII码
        int right = 0;
        int left = 0;
        int begin = 0;
        int end = 0;
        int minlen = sLen+1;
        int distance = 0;//记录滑动窗口中有多少个目标字符,若某个字符超过需要值我们不计入distance中
        while(right < sLen)
        {
            char charRight = ss[right];//滑动窗口右边的字符
            if(tAll[charRight]==0)//此字符不属于目标字符串中字符,滑动窗口右边右移
            {
                right++;
                continue;
            }
            if(win[charRight]<tAll[charRight])//属于目标且滑动窗口刚好欠缺(超出了便不计入distance)
            {
                distance++; 
            }
            win[charRight]++;
            right++;
            while(distance==tLen)//滑动窗口包含了全部目标字符串(甚至有的字符会超出需要)
            {
                if(right-left<minlen)//更新minlen,begin,end
                {
                    minlen = right-left;
                    begin = left;
                    end = right;
                }
                char charLeft = ss[left];//滑动窗口左边的字符
                if(tAll[charLeft]==0)//此字符不属于目标,滑动窗口左边右移
                {
                    left++; continue;
                }
                if(win[charLeft]==tAll[charLeft])//此字符频度处于临界值,滑动窗口左边右移,此时滑动窗口不符合条件,等待滑动窗口右边右移再次符合条件
                {
                    distance--;
                }
                win[charLeft]--;//就算滑动窗口左边存在某个字符超出需要,此时也可以向右滑动并更新minlen,begin,end
                left++;
            }
        }
        if(minlen==sLen+1) return "";
        return s.substring(begin,end);
    }

以上理解均基于leetcode的题解,详见:
leetcode 76. 最小覆盖子串 题解

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
1. 二分法 5 1.1. 什么是二分查找 5 1.2. 如何识别二分法 5 1.3. 二分法模板 6 1.3.1. 模板一 6 1.3.1.1. 模板代码 6 1.3.1.2. 关键属性 7 1.3.1.3. 语法说明 7 1.3.1.4. Lc69:x的平方根 8 1.3.1.5. Lc374:猜数大小 9 1.3.1.6. Lc33:搜索旋转数组 11 1.3.2. 模板二 13 1.3.2.1. 模板代码 13 1.3.2.2. 关键属性 14 1.3.2.3. 语法说明 14 1.3.2.4. Lc278:第一个错误版本 14 1.3.2.5. Lc162:寻找峰值 16 1.3.2.6. Lc153:寻找旋转排序数组最小值 19 1.3.2.7. Lc154:寻找旋转排序数组最小值II 20 1.3.3. 模板三 22 1.3.3.1. 模板代码 22 1.3.3.2. 关键属性 23 1.3.3.3. 语法说明 23 1.3.3.4. LC-34:在排序数组中查找元素的第一个和最后一个 23 1.3.3.5. LC-658:找到K个最接近的元素 25 1.3.4. 小结 28 1.4. LeetCode中二分查找题目 29 2. 双指针 30 2.1. 快慢指针 31 2.1.1. 什么是快慢指针 31 2.1.2. 快慢指针模板 31 2.1.3. 快慢指针相关题目 32 2.1.3.1. LC-141:链表是否有环 32 2.1.3.2. LC-142:环形链表入口 34 2.1.3.3. LC-876:链表的中间节点 37 2.1.3.4. LC-287:寻找重复数 40 2.2. 滑动窗口 43 2.2.1. 什么是滑动窗口 43 2.1.4. 常见题型 44 2.1.5. 注意事项 45 2.1.6. 滑动窗口模板 45 2.1.7. 滑动窗口相关题目 46 2.1.7.1. LC-3:无重复字符的最长子串 47 2.1.7.2. LC-76:最小覆盖子串 49 2.1.7.3. LC-209:长度最小的子数组 54 2.1.7.4. LC-239:滑动窗口最大值 57 2.1.7.5. LC-395:至少有K个重复字符的最长子串 60 2.1.7.6. LC-567:字符串排列 62 2.1.7.7. LC-904:水果成篮 64 2.1.7.8. LC-424:替换后的最长重复字符 66 2.1.7.9. LC-713:乘积小于K的子数组 67 2.1.7.10. LC-992:K个不同整数的子数组 70 2.3. 左右指针 73 2.3.1. 模板 73 2.3.2. 相关题目 73 2.3.2.1. LC-76:删除倒数第N个节点 74 2.3.2.2. LC-61:旋转链表 76 2.3.2.3. LC-80:删除有序数组中的重复项 79 2.3.2.4. LC-86:分割链表 80 2.3.2.5. LC-438:找到字符串中所有字母的异位词 82 3. 模板 85 2.3.2.6. LC-76:删除倒数第N个节点 85

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值