leetcode 76 最小覆盖子集1.
题目描述
给定一个字符串 S 和一个字符串 T,请在 S 中找出包含 T 所有字母的最小子串。
示例:
输入:s=”ADOBECODEBANC”, t=”ABC”
输出:”BNAC”
说明:
- 如果 S 中不存这样的子串,则返回空字符串 “”。
- 如果 S 中存在这样的子串,我们保证它是唯一的答案。
解题思路
此题采用滑动窗口的方法,用左右两个位标维持一个最小窗口,使其包含t中所有字符。扫描一遍后给出最小窗口的左右位标,时间复杂度为O(n)。
本文采用两个数组ms[256]和mt[256]来存放字符在s,v中出现的次数。其空间复杂度为ASCII码的长度。
主要步骤
(1)s中维护一个滑动窗口[l.r],初始时,l=r=0;不断扩大右边界使[l,r]中包含所有t中元素,用一个计数器count计数。直到r=s.size() 转(4);
(2)当count==t.size()时,缩小左边界,使[l,r]为此时包含t中全部元素切区间最小;
(3)与最小区间值minW比较并进行替换,移动左边界,count–,转(1);
(4)返回最小窗口区间中的元素,若无则返回空字符串。
代码
class Solution {
//时间复杂度O(n)
//空间复杂度O(ascii.size())
public:
string minWindow(string s, string t) {
int ms[256]={0};
int mt[256]={0};
int l = 0;
int minl = 0;
int minW = s.size()+1;
int count = 0;
string res = "";
if(t.size() == 0) return res;
for(int i = 0; i < t.size();++i)
mt[t[i]]++;
for(int r = 0; r < s.size(); r++){
if(count < t.size()){//[l,r]中不含有t中全部元素r++
if(mt[s[r]] > ms[s[r]]) count++;
ms[s[r]]++;
if(count<t.size()) continue;
}
if(count == t.size()){//[l,r]中含有t中全部元素
while(mt[s[l]] < ms[s[l]]){//缩小左边界
ms[s[l]]--;
l++;
}
if(minW > r -l + 1){//判断当前区间[l,r]是否为最小区间,并替换
minW = r -l + 1;
minl = l;
}
res = s.substr(minl,minW);
//在[l+1...)中继续寻找含有t中全部元素区间
ms[s[l++]]--;
count--;
}
}
return res;
}
};
目录
[1]https://leetcode-cn.com/problems/minimum-window-substring/description/
- 这里是 脚注 的 内容. ↩