Leetcode76覆盖最小子串

覆盖最小子串

  • 代码来自b站左程云
java 复制代码
class Solution {
    public String minWindow(String str, String tar) {
        char[] s = str.toCharArray();
        char[] t = tar.toCharArray();
        int[] cnt = new int[256];
        for (char cha : t) { 
            cnt[cha]--;
        }
        int len = Integer.MAX_VALUE;
        int debt = t.length;
        int start = 0;
        for (int r = 0, l = 0; r < s.length; r ++) { 
            if (cnt[s[r]]++ < 0) { 
                debt--;
            }
            if (debt == 0) { 
                while (cnt[s[l]] > 0) { 
                    cnt[s[l++]]--;
                }
                if (r - l + 1 < len) { 
                    len = r - l + 1;
                    start = l;
                }
            }

        }
        return len == Integer.MAX_VALUE ? "" : str.substring(start, start + len);
    }
}

画图理解题意

  • 我们先梳理一下思路:
  1. 我们要确定这个窗口有没有包含target字符串中的每一个字符,难道我们要遍历比较吗?显然不行,那么怎么样让它遍历一次就知道是否包含呢?
  2. 我们利用之前前缀和中哈希表的思路,我们把target字符串的每个字符出现的次数作为每个字符欠债的个数存到数组中,把它弄成一个前债表。
  • 我们看第一个样例:

初始的欠债表为:

说明此时我们要找到一个满足ABC每个字符各一个的组合。


当我们不断扩展右边界,会发现第一次满足条件的窗口是这样的:

可是题目要我们求最短啊,我们尝试收缩左边界,收缩的时候要注意,如果收缩会导致欠债那么就不能收缩,只能记住答案,拿去与之前比大小看是否能更新。

然后继续扩展右边界:

此时我们发现,不仅不欠债还有了结余可以尝试收缩。

继续这样推下去,就是不断进行这个判断过程:

理解代码:

java 复制代码
    if (cnt[s[r]]++ < 0) { 
         debt--;
    }

这里是把每一个字符扔到欠债表里面进行结算,如果是target里面的,说明他刚开始是负的,所以我们用是否小于0来判断是否可以对欠债总数debt来进行削减,到0的时候就说明我们要开始尝试收缩窗口了。

java 复制代码
          if (debt == 0) { 
                while (cnt[s[l]] > 0) { 
                    cnt[s[l++]]--;
                }
                if (r - l + 1 < len) { 
                    len = r - l + 1;
                    start = l;
                }
            }

要判断左边是否可以削减,就要看它的削减会不会导致债务的增加,也就是会不会导致现在的窗口不能完全包含target,所以进行此判断。

然后我们就要看这个窗口长度是不是目前最短的,是的话就更新它,同时记住此时左边界,为什么?因为我们要返回的是一个字符串,截取一个子串需要它的长度和起始位置。

相关推荐
CCCC13101631 分钟前
嵌入式学习(day 28)线程
jvm·学习
星星火柴9361 小时前
关于“双指针法“的总结
数据结构·c++·笔记·学习·算法
小狗爱吃黄桃罐头1 小时前
正点原子【第四期】Linux之驱动开发篇学习笔记-1.1 Linux驱动开发与裸机开发的区别
linux·驱动开发·学习
艾莉丝努力练剑2 小时前
【洛谷刷题】用C语言和C++做一些入门题,练习洛谷IDE模式:分支机构(一)
c语言·开发语言·数据结构·c++·学习·算法
武昌库里写JAVA3 小时前
JAVA面试汇总(四)JVM(一)
java·vue.js·spring boot·sql·学习
C++、Java和Python的菜鸟3 小时前
第六章 统计初步
算法·机器学习·概率论
Cx330❀3 小时前
【数据结构初阶】--排序(五):计数排序,排序算法复杂度对比和稳定性分析
c语言·数据结构·经验分享·笔记·算法·排序算法
杜子不疼.3 小时前
《Python学习之字典(一):基础操作与核心用法》
开发语言·python·学习
散1123 小时前
01数据结构-Prim算法
数据结构·算法·图论
小幽余生不加糖4 小时前
电路方案分析(二十二)适用于音频应用的25-50W反激电源方案
人工智能·笔记·学习·音视频