力扣HOT100之子串:76. 最小覆盖子串

这道题目只写出来一半,我想着思路和之前那个无重复字符的最长子串有点像,然后就依葫芦画瓢写了一部分,但是还是没有想到额外定义一个左右指针来维护最短子串的左右端点的下标,这里把这个思路具体说下。

首先我们依然需要定义两个哈希表hash_Thash_S,其中hash_T用于统计字符串t中各个字符的分布情况,而hash_S用于统计滑动窗口内的字符分布情况。首先,不管字符串s能否涵盖字符串t,我们都默认s能涵盖t,因此我们定义result_leftresult_right分别代表最终的最短子串的起点下标和终点下标,在初始状态下,令result_left = -1result_right = s.size(),然后我们再定义左右指针leftright,两个指针在初始状态下都指向s的第一个字符,然后使用for循环用右指针遍历s中的字符,然后我们先将右指针指向的字符计入hash_S,然后判断hash_S是否涵盖了hash_T(hash_T中的每一种字符hash_S中都要有,而且hash_T中每一种字符的个数不大于hash_S中对应的字符个数),如果涵盖了,就说明当前滑动窗口已经找到了符合要求的子串,如果该子串的长度(right - left)小于上一个符合要求的子串的长度(result_right - result_left)直接将left赋值给result_left,将right赋值给result_right,这就实现了结果的更新。注意,在更新结果后需要及时将left指向的字符的数量-1,并将left右移。只要hash_S涵盖了hash_T,就一直循环记录结果,当不再涵盖时,再进行下一次for循环。当外层的for循环结束后,我们需要判断s是否真的满足涵盖条件,如果自始至终都不满足,那么result_left将一直指向-1,此时直接返回空字符串即可,否则就返回s字符串中[result_left, result_right]区间范围内的子串。

cpp 复制代码
class Solution {
public:
    //判断
    bool is_covered(unordered_map<char, int>& hash_T, unordered_map<char, int>& hash_S){
        for(pair<const char, int>& p : hash_T){
            if(!hash_S.contains(p.first) || hash_S[p.first] < hash_T[p.first])
                return false;
        }
        return true;
    }
    string minWindow(string s, string t) {
        int result_left = -1, result_right = s.size();
        unordered_map<char, int> hash_T;    //用来存储字符串t的字符分布情况
        unordered_map<char, int> hash_S;    //用来存储子串内的字符分布情况
        for(char& c : t)
            hash_T[c]++;
        for(int left = 0, right = 0; right < s.size(); right++){
            hash_S[s[right]]++;
            while(is_covered(hash_T, hash_S)){  //当t涵盖s时执行循环
                if(right - left < result_right - result_left){  //寻找到更短的子串
                    result_left = left;
                    result_right = right;
                }
                hash_S[s[left]]--;
                left++;
            }
        }
        return result_left < 0 ? "" : s.substr(result_left, result_right - result_left + 1);
    }
};
相关推荐
零小陈上(shouhou6668889)3 分钟前
K-近邻算法 - lazy learning的代表
算法·近邻算法
有一个好名字9 分钟前
力扣-从字符串中移除星号
java·算法·leetcode
萧瑟其中~13 分钟前
二分算法模版——基础二分查找,左边界查找与右边界查找(Leetcode的二分查找、在排序数组中查找元素的第一个位置和最后一个位置)
数据结构·算法·leetcode
码上就好ovo15 分钟前
Atcoder Beginnner Contest 440
算法
高洁0122 分钟前
CLIP 的双编码器架构是如何优化图文关联的?(3)
深度学习·算法·机器学习·transformer·知识图谱
jllllyuz28 分钟前
MATLAB实现蜻蜓优化算法
开发语言·算法·matlab
AlenTech29 分钟前
208. 实现 Trie (前缀树) - 力扣(LeetCode)
leetcode
iAkuya30 分钟前
(leetcode)力扣100 36二叉树的中序遍历(迭代递归)
算法·leetcode·职场和发展
wangwangmoon_light38 分钟前
1.1 LeetCode总结(线性表)_枚举技巧
算法·leetcode·哈希算法