力扣Hot100题目解析:最小覆盖子串(滑动窗口+哈希表详细注释)

力扣Hot100题目解析:最小覆盖子串(滑动窗口+哈希表详细注释)

一、题目简介

题目名称:最小覆盖子串(Minimum Window Substring)

题目描述: 给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串,则返回空字符串 "" 。

示例:

输入:s = "ADOBECODEBANC", t = "ABC"
输出:"BANC"

二、解题思路

本题的核心是用滑动窗口+哈希表统计来高效查找。详细思路如下:

  1. 统计目标字符需求:用一个Map统计字符串t中每个字符需要出现的次数。
  2. 滑动窗口定义:用两个指针leftright表示当前窗口区间。
  3. 扩展窗口:不断移动right指针,遇到t需要的字符就减少需求。
  4. 窗口收缩:当窗口已经包含所有t中的字符,尝试移动left指针收缩窗口,并维护最小长度。
  5. 更新结果:每次窗口满足条件时,更新记录的最小区间。
  6. 输出最小子串:遍历结束返回记录的最小区间对应的子串。

这种方法时间复杂度为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,算法,字符串,面试题

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值