【 每天学习一点算法 2026/05/08】最小覆盖子串

每天学习一点算法 2026/05/08

题目:最小覆盖子串

给定两个字符串 s 和 t,长度分别是 m 和 n,返回 s 中的 最短窗口 子串,使得该子串包含 t 中的每一个字符(包括重复字符)。如果没有这样的子串,返回空字符串 ""。

测试用例保证答案唯一。

作者:LeetCode

链接:https://leetcode.cn/leetbook/read/top-interview-questions-hard/xw1tws/

来源:力扣(LeetCode)

著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

这个题需要用到滑动窗口,我们可以维护两个指针 leftright 分别指向窗口两个端点,循环判断当窗口内的子串能覆盖目标值时,left 右移,当窗口内子串不能覆盖目标值时, right右移,每次满足覆盖条件式记录最小值,直到 right 指向 s 的最右侧且窗口内子串不满足覆盖条件时,结束循环。

由于字符串 t 可能会有重复字符,所以我们需要维护两个 map 来统计窗口内字符串的字符数量和目标字符串 t 的字符数量。

typescript 复制代码
function minWindow(s: string, t: string): string {
  // 检查当前窗口是否满足条件
  function check() {
    for (const [char, count] of mapT) {
      if (!mapS.has(char) || mapS.get(char)! < count) {
        return false;
      }
    }
    return true;
  }

  const mapS = new Map<string, number>(); // 窗口内字符计数
  const mapT = new Map<string, number>(); // t 字符计数

  // 初始化 t 的频率表
  for (const char of t) {
    mapT.set(char, (mapT.get(char) || 0) + 1);
  }

  let left = 0, min = Infinity, startL = -1;

  // 右指针遍历
  for (let right = 0; right < s.length; right++) {
    // 不断移动右指针并记录窗口内字符频率表
    const charR = s[right];
    mapS.set(charR, (mapS.get(charR) || 0) + 1);

    // 满足条件就收缩左指针
    while (check()) {
      // 更新最小窗口
      if (right - left + 1 < min) {
        min = right - left + 1;
        startL = left;
      }
      // 左指针右移
      const charL = s[left];
      mapS.set(charL, mapS.get(charL)! - 1);
      left++;
    }
  }

  return startL === -1 ? '' : s.substring(startL, startL + min);
}

上面的这个方法时间复杂度高主要是 check 函数造成的,因为每次都需要遍历一次 mapT 来判断是否满足条件,这里我们其实可以用维护一个变量 match 来统计 窗口中字符匹配目标字符串的字符及其个数的,也就是 t 中的字符在 s 中存在且数量一致时 match 数量才加一,当 match 和 mapT.size 相等时就表示满足覆盖条件了,这样我们就可以进一步优化时间复杂度

typescript 复制代码
function minWindow(s: string, t: string): string {
  const need = new Map<string, number>(); // 存储需要匹配的字符及其个数
  const window = new Map<string, number>(); // 存储窗口中的字符及其个数

  // 统计 t 中字符及其个数
  for (const c of t) need.set(c, (need.get(c) || 0) + 1);

  let left = 0, right = 0;
  let match = 0; // 匹配成功的字符种类数
  let start = 0, minLen = Infinity;
  
  // 循环直至右指针到达字符串 s 右边界
  while (right < s.length) {
    const c = s[right]; // 记录窗口新增字符
    right++; // 移动有指针

    // 更新窗口内字符统计
    if (need.has(c)) {
      window.set(c, (window.get(c) || 0) + 1);
      // 如果当前字符满足匹配条件 match 加一
      if (window.get(c) === need.get(c)) match++;
    }

    // 当 match 和 need.size 相等表示满足覆盖条件,收缩窗口
    while (match === need.size) {
      // 更新最小窗口
      if (right - left < minLen) {
        minLen = right - left;
        start = left;
      }

      const d = s[left];
      left++;

      // 更新窗口内字符统计
      if (need.has(d)) {
        // 如果是需匹配字符串的字符需要更新 match
        if (window.get(d) === need.get(d)) match--;
        // 窗口内该字符数量减一
        window.set(d, window.get(d)! - 1);
      }
    }
  }

  return minLen === Infinity ? '' : s.substr(start, minLen);
}

题目来源:力扣(LeetCode)

相关推荐
汉克老师2 小时前
GESP5级C++考试语法知识(十六、分治算法(三))
c++·算法·分治算法·汉诺塔·逆序对·gesp5级·gesp五级
南境十里·墨染春水2 小时前
linux学习进展 I/O复用函数初步
linux·运维·学习
V搜xhliang02462 小时前
OpenClaw进阶完全教程
运维·人工智能·算法·microsoft·自动化
叼烟扛炮2 小时前
C++ 知识点12 构造函数
开发语言·c++·算法·构造函数
开开心心就好2 小时前
支持音视频图片文档的格式转换器
人工智能·学习·游戏·决策树·音视频·动态规划·语音识别
2501_940655972 小时前
Paperiii 官网入口:www.paperiii.com——2026抖音爆款AI写作工具
人工智能·学习·ai写作
我想我不够好。2 小时前
2026.5.9消防监控学习 40min
学习
羽沢312 小时前
Canvas学习一
前端·css·学习·canvas
YouCanYouUp.2 小时前
从硬件中断到软件回调:深入理解中断向量表设计与实践
mcu·学习