博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
[LeetCode] Minimum Window Substring 最小窗口子串
阅读量:6424 次
发布时间:2019-06-23

本文共 3525 字,大约阅读时间需要 11 分钟。

 

Given a string S and a string T, find the minimum window in S which will contain all the characters in T in complexity O(n).

Example:

Input: S = "ADOBECODEBANC", T = "ABC"Output: "BANC"

Note:

  • If there is no such window in S that covers all characters in T, return the empty string "".
  • If there is such window, you are guaranteed that there will always be only one unique minimum window in S.

 

这道题给了我们一个原字符串S,还有一个目标字符串T,让我们在S中找到一个最短的子串,使得其包含了T中的所有的字母,并且限制了时间复杂度为O(n)。这道题的要求是要在O(n)的时间度里实现找到这个最小窗口字串,那么暴力搜索Brute Force肯定是不能用的,因为遍历所有的子串的时间复杂度是平方级的。那么我们想,时间复杂度卡的这么严,说明我们必须在一次遍历中完成任务,当然遍历若干次也是O(n),但不一定有这个必要,尝试就一次遍历拿下!那么再来想,既然既然要包含T中所有的字母,那么肯定对于T中的每个字母,肯定要快速查找是否在子串中,既然总时间都卡在了O(n),我们肯定不想在这里还浪费时间,那么就空间换时间(也就算法题中可以这么干了,七老八十的富翁就算用大别野也换不来时间啊。依依东望,望的就是时间呐T.T),使用HashMap,建立T中每个字母与其出现次数之间的映射,那么你可能会有疑问,为啥不用HashSet呢,别急,讲到后面你就知道用HashMap有多妙,简直妙不可言~

目前在脑子一片浆糊的情况下,我们还是从简单的例子来分析吧,题目例子中的S有点长,我们换个短的 S = "ADBANC",T = "ABC",那么我们肉眼遍历一遍S呗,首先第一个是A,嗯很好,T中有,第二个是D,T中没有,不理它,第三个是B,嗯很好,T中有,第四个又是A,多了一个,礼多人不怪嘛,收下啦,第五个是N,一边凉快去,第六个终于是C了,那么貌似好像需要整个S串,其实不然,我们注意之前有多一个A,但么我们就算去掉第一个A,也没事,因为第四个A可以代替之,第二个D也可以去掉,因为不在T串中,第三个B就不能再去掉了,不然就没有B了。所以最终的答案就"BANC"了。通过上面的描述,你有没有发现一个有趣的现象,我们是先扩展,再收缩,就好像一个窗口一样,先扩大右边界,然后再收缩左边界,上面的例子中我们是右边界无法扩大了后才开始收缩左边界,实际上对于复杂的例子,有可能是扩大右边界,然后缩小一下左边界,然后再扩大右边界等等。这就很像一个不停滑动的窗口了,这就是大名鼎鼎的滑动窗口Sliding Window了,简直是神器啊,能解很多子串,子数组,子序列等等的问题,是必须要熟练掌握的啊!

下面我们来考虑用代码来实现,先来回答一下前面埋下的伏笔,为啥要用HashMap,而不是HashSet,现在应该很显而易见了吧,因为要统计T串中字母的个数,而不是仅仅看某个字母是否在T串中出现。统计好T串中字母的个数了之后,我们开始遍历S串,对于S中的每个遍历到的字母,都在HashMap中的映射值减1,如果减1后的映射值仍大于等于0,说明当前遍历到的字母是T串中的字母,我们使用一个计数器cnt,使其自增1。当cnt和T串字母个数相等时,说明此时的窗口已经包含了T串中的所有字母,此时更新一个minLen和结果res,这里的minLen是我们维护的一个全局变量,用来记录出现过的包含T串所有字母的最短的子串的长度,结果res就是这个最短的子串。然后我们要开始收缩左边界,由于我们遍历的时候,对映射值减了1,所以此时去除字母的时候,就要把减去的1加回来,此时如果加1后的值大于0了,说明此时我们少了一个T中的字母,那么cnt值就要减1了,然后移动左边界left。那么你可能会疑问,对于不在T串中的字母的映射值也这么加呀减呀的,真的大丈夫(带脚布)吗?其实没啥事,因为对于不在T串中的字母,我们减1后,变-1,cnt不会增加,之后收缩左边界的时候,映射值加1后为0,cnt也不会减少,所以并没有什么影响啦,下面是具体的步骤啦:

- 我们最开始先扫描一遍T,把对应的字符及其出现的次数存到HashMap中。

- 然后开始遍历S,就把遍历到的字母对应的HashMap中的value减一,如果减1后仍大于等于0,cnt自增1。

- 如果cnt等于T串长度时,开始循环,纪录一个字串并更新最小字串值。然后将子窗口的左边界向右移,如果某个移除掉的字母是T串中不可缺少的字母,那么cnt自减1,表示此时T串并没有完全匹配。

 

解法一:

class Solution {public:    string minWindow(string s, string t) {        string res = "";        unordered_map
letterCnt; int left = 0, cnt = 0, minLen = INT_MAX; for (char c : t) ++letterCnt[c]; for (int i = 0; i < s.size(); ++i) { if (--letterCnt[s[i]] >= 0) ++cnt; while (cnt == t.size()) { if (minLen > i - left + 1) { minLen = i - left + 1; res = s.substr(left, minLen); } if (++letterCnt[s[left]] > 0) --cnt; ++left; } } return res; }};

 

这道题也可以不用HashMap,直接用个int的数组来代替,因为ASCII只有256个字符,所以用个大小为256的int数组即可代替HashMap,但由于一般输入字母串的字符只有128个,所以也可以只用128,其余部分的思路完全相同,虽然只改了一个数据结构,但是运行速度提高了一倍,说明数组还是比HashMap快啊,代码如下:

 

解法二:

class Solution {public:    string minWindow(string s, string t) {        string res = "";        vector
letterCnt(128, 0); int left = 0, cnt = 0, minLen = INT_MAX; for (char c : t) ++letterCnt[c]; for (int i = 0; i < s.size(); ++i) { if (--letterCnt[s[i]] >= 0) ++cnt; while (cnt == t.size()) { if (minLen > i - left + 1) { minLen = i - left + 1; res = s.substr(left, minLen); } if (++letterCnt[s[left]] > 0) --cnt; ++left; } } return res; }};

 

类似题目:

 

参考资料:

 

转载地址:http://tsrra.baihongyu.com/

你可能感兴趣的文章
《从零构建前后分离的web项目》实战 -5分钟快速构建炒鸡规范的VUE项目骨架
查看>>
【wol】远程开机
查看>>
Blender Python API概述
查看>>
css 动画
查看>>
关于使用别人给的已安装好create-react-app的webstorm空项目
查看>>
SpringBoot非官方教程 | 第二十一篇: springboot集成JMS
查看>>
WEEX-android播放背景音乐 (使用module)
查看>>
linux下安装node npm并配置
查看>>
基于asyncio编写一个telegram爬虫机器人
查看>>
vue.js+socket.io+express+mongodb打造在线聊天室[二]
查看>>
parcel打包vue
查看>>
什么是Scrum
查看>>
nginx负载均衡的5种策略
查看>>
90%人都不知道:SVN 和 Git 的一些误解和真相
查看>>
基于Vue-Cli的非跨域请求模拟数据(Mock)快速配置更新接口的解决方案
查看>>
赛事直播解说+连麦,技术架构难点解读
查看>>
keystonejs实战之页头页脚
查看>>
8.注册和登录功能实现(1)—— 页面设计和获取POST数据
查看>>
【358天】每日项目总结系列095(2018.01.29)
查看>>
Angular 5.0 学习1:Angular 5.0介绍
查看>>