leetcode 76

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(countsize()) 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;
    }
};

目录

  • leetcode 76 最小覆盖子集1.
      • 题目描述
      • 解题思路
      • 代码
      • 目录

[1]https://leetcode-cn.com/problems/minimum-window-substring/description/


  1. 这里是 脚注内容. ↩

你可能感兴趣的:(leetcode)