力扣Hot100题目解析:最小覆盖子串(滑动窗口+哈希表详细注释)
一、题目简介
题目名称:最小覆盖子串(Minimum Window Substring)
题目描述: 给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串,则返回空字符串 "" 。
示例:
输入:s = "ADOBECODEBANC", t = "ABC"
输出:"BANC"
二、解题思路
本题的核心是用滑动窗口+哈希表统计来高效查找。详细思路如下:
- 统计目标字符需求:用一个
Map
统计字符串t
中每个字符需要出现的次数。 - 滑动窗口定义:用两个指针
left
和right
表示当前窗口区间。 - 扩展窗口:不断移动
right
指针,遇到t
需要的字符就减少需求。 - 窗口收缩:当窗口已经包含所有
t
中的字符,尝试移动left
指针收缩窗口,并维护最小长度。 - 更新结果:每次窗口满足条件时,更新记录的最小区间。
- 输出最小子串:遍历结束返回记录的最小区间对应的子串。
这种方法时间复杂度为O(n),空间复杂度为O(|t|)。
三、Java详细实现(带注释)
import java.util.HashMap;
import java.util.Map;
public class MinWindowSubstring {
/**
* 返回s中包含t所有字符的最小子串
* @param s 源字符串
* @param t 目标字符集
* @return 最小覆盖子串
*/
public String minWindow(String s, String t) {
if (s.length() < t.length()) return "";
// 统计t中每个字符的需求
Map<Character, Integer> need = new HashMap<>();
for (char c : t.toCharArray()) {
need.put(c, need.getOrDefault(c, 0) + 1);
}
// 滑动窗口内各字符出现次数
Map<Character, Integer> window = new HashMap<>();
int left = 0, right = 0; // 窗口左右指针
int valid = 0; // 有效字符个数
int start = 0, minLen = Integer.MAX_VALUE; // 结果记录
while (right < s.length()) {
char c = s.charAt(right);
right++; // 扩大窗口
// 更新窗口数据
if (need.containsKey(c)) {
window.put(c, window.getOrDefault(c, 0) + 1);
if (window.get(c).intValue() == need.get(c).intValue()) {
valid++;
}
}
// 当窗口包含所有t中的字符,尝试收缩
while (valid == need.size()) {
// 更新最小区间
if (right - left < minLen) {
start = left;
minLen = right - left;
}
char d = s.charAt(left);
left++; // 收缩窗口
if (need.containsKey(d)) {
if (window.get(d).intValue() == need.get(d).intValue()) {
valid--;
}
window.put(d, window.get(d) - 1);
}
}
}
return minLen == Integer.MAX_VALUE ? "" : s.substring(start, start + minLen);
}
// 测试用例
public static void main(String[] args) {
MinWindowSubstring solution = new MinWindowSubstring();
String s = "ADOBECODEBANC";
String t = "ABC";
String result = solution.minWindow(s, t);
System.out.println("最小覆盖子串: " + result);
}
}
四、总结
滑动窗口结合哈希表是处理字符串区间问题的经典技巧。这类题型要学会窗口的扩展与收缩,以及如何判定窗口状态。
文章标签:力扣,Hot100,最小覆盖子串,滑动窗口,哈希表,Java,算法,字符串,面试题