news 2026/4/16 5:48:02

双指针专题(八):步长跳跃的艺术——「串联所有单词的子串」

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
双指针专题(八):步长跳跃的艺术——「串联所有单词的子串」

场景想象:

你有一串很长的珍珠项链(字符串 s),和一堆散落的、长度相同的宝石(单词数组 words)。

你需要从项链上截取一段,使得这段子串 恰好 由所有的宝石串联而成(顺序不限,但每个宝石都要用上,且不能多不能少)。

难点升级:

  1. 步长变化:以前left++right++是挪动 1 个字符。现在,因为单词长度固定为k,我们每次必须挪动k个字符

  2. 分组遍历:如果我们只从index=0开始每次跳k步,我们会漏掉从index=1开始的情况。所以我们需要错位触发

力扣 30. 串联所有单词的子串

https://leetcode.cn/problems/substring-with-concatenation-of-all-words/

题目分析:

  • 输入:字符串s,字符串数组wordswords里所有单词长度相同,记为oneWordLen)。

  • 目标:在s中找到所有起始索引,使得从该位置开始的子串恰好包含words中所有单词各一次。

  • 输出:索引数组。

核心思维:多路滑动窗口 + HashMap 计数

这其实就是 LC 438 异位词 的“巨人版”。

我们将 s 看作是一个由“单词”组成的数组。

假设单词长度 oneWordLen = 3。

我们需要分别从 0, 1, 2 这三个偏移量开始跑滑动窗口。

  • Offset 0: 检查s[0...2], s[3...5], s[6...8]...

  • Offset 1: 检查s[1...3], s[4...6], s[7...9]...

  • Offset 2: 检查s[2...4], s[5...7], s[8...10]...

这样就能覆盖所有可能的切分情况。

代码实现 (JavaScript)

JavaScript

/** * @param {string} s * @param {string[]} words * @return {number[]} */ var findSubstring = function(s, words) { if (!s || !words || words.length === 0) return []; const oneWordLen = words[0].length; const wordCount = words.length; // 总长度:所有单词拼起来有多长 const allWordsLen = oneWordLen * wordCount; const res = []; // 1. 统计 words 里的单词需求 const need = new Map(); for (const w of words) { need.set(w, (need.get(w) || 0) + 1); } // 2. 核心:外层循环控制“偏移量” (Offset) // 只要遍历 0 到 oneWordLen - 1 即可覆盖所有情况 // 比如单词长 3,我们只需要从 0, 1, 2 开始跑三次滑动窗口 for (let i = 0; i < oneWordLen; i++) { let left = i; let right = i; let count = 0; // 窗口内有效匹配的单词个数 const window = new Map(); // 当前窗口里的单词统计 // 内层循环:标准的滑动窗口,每次跳 oneWordLen 步 while (right + oneWordLen <= s.length) { // --- 进窗口 --- // 截取一个单词 const w = s.substring(right, right + oneWordLen); right += oneWordLen; // 指针大步跃进 // 如果这个单词不在需求列表里,说明这一段全废了 if (!need.has(w)) { // 直接清空窗口,从下一个位置重新开始 window.clear(); count = 0; left = right; } else { // 单词是有效的,加入窗口 window.set(w, (window.get(w) || 0) + 1); count++; // --- 出窗口 --- // 如果窗口里某个单词的数量超过了需求 // 比如 need={A:1},window={A:2},这时候必须缩左边,直到把多余的 A 吐出去 while (window.get(w) > need.get(w)) { const leftWord = s.substring(left, left + oneWordLen); window.set(leftWord, window.get(leftWord) - 1); count--; left += oneWordLen; // 左指针也大步跃进 } // --- 检查结果 --- // 如果单词数量够了,记录起始位置 if (count === wordCount) { res.push(left); } } } } return res; };

深度解析:为什么需要外层循环i

这是这道题最难理解的点。

假设 s = "barfoothefoobarman", words = ["foo", "bar"] (长度3)。

  • 如果不加外层循环,直接从 0 开始滑:

    • 检查bar,foo,the... (索引 0, 3, 6...)

  • 但是,如果答案是从索引 1 开始的呢?(虽然这题不是,但假设s = "afoo..."

    • 从 0 开始滑会读到afo,o...,完全错位。

  • 所以,我们需要:

    • i=0: 处理索引0, 3, 6, 9...的单词切分。

    • i=1: 处理索引1, 4, 7, 10...的单词切分。

    • i=2: 处理索引2, 5, 8, 11...的单词切分。

    • i=3: 不需要了,因为它和i=0是重合的(只是少了一个单词)。

总结

这道题是 LC 438 的进阶版,也是 LC 76 的变种。

它强迫你把“字符”的微观视角,转换成“单词块”的宏观视角。

  • 关键点 1:外层循环遍历0wordLen - 1,消除“相位差”。

  • 关键点 2:内层循环的步长是wordLen

  • 关键点 3:遇到非法单词直接window.clear(),这是极大的优化(剪枝)。


下一题预告:滑动窗口最大值

拿下了这道“步长跳跃”的难题,下一关我们要去挑战一个需要特殊武器的题目——LC 239. 滑动窗口最大值(Hard)。

  • 题目很简单:一个窗口滑过数组,每次告诉你窗口里最大的那个数是谁。

  • 陷阱:如果你每次遍历窗口找最大值,时间复杂度是 $O(N \times K)$,会超时。

  • 新武器:你需要学会使用“单调队列” (Monotonic Queue)。这是一种虽然叫“队列”,但里面的人必须按身高排队的神奇结构。

准备好解锁新数据结构了吗?🚀

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/11 4:18:40

清华镜像站使用教程:一键拉取HunyuanOCR Docker镜像

清华镜像站加速部署 HunyuanOCR&#xff1a;一条命令跑通国产端到端 OCR 在智能文档处理需求激增的今天&#xff0c;企业与开发者对OCR技术的要求早已不止“识别文字”这么简单。面对复杂版式、多语言混排、字段精准抽取等现实挑战&#xff0c;传统级联式OCR方案越来越显得力不…

作者头像 李华
网站建设 2026/4/14 15:29:23

UltraISO引导U盘制作含HunyuanOCR Linux系统的可行性

UltraISO引导U盘制作含HunyuanOCR Linux系统的可行性 在政府档案数字化现场&#xff0c;一名工作人员将U盘插入老旧台式机——这台设备既无管理员权限&#xff0c;也未安装任何AI框架。30秒后&#xff0c;系统自动启动一个轻量Linux环境&#xff0c;浏览器弹出HunyuanOCR的Web界…

作者头像 李华
网站建设 2026/4/14 8:52:40

JavaScript Base64编码图片上传至HunyuanOCR接口

JavaScript Base64编码图片上传至HunyuanOCR接口 在智能办公和文档数字化浪潮席卷各行各业的今天&#xff0c;用户对“拍一下就能识别文字”的体验早已习以为常。无论是扫描合同、翻译外文标签&#xff0c;还是从身份证中提取信息&#xff0c;背后都离不开OCR技术的支持。但如何…

作者头像 李华
网站建设 2026/4/15 18:55:33

GitHub镜像网站推荐列表:稳定获取HunyuanOCR及其他AI模型

GitHub镜像网站推荐&#xff1a;高效获取HunyuanOCR等AI模型的实用指南 在当前AI技术快速落地的大背景下&#xff0c;开发者最常遇到的一个“小问题”却可能成为项目推进的“大瓶颈”——如何稳定、快速地下载托管在GitHub上的大型AI模型&#xff1f;尤其是像腾讯推出的Hunyuan…

作者头像 李华
网站建设 2026/4/5 19:41:20

计算机毕业设计springboot大学生心理健康咨询系统 基于Spring Boot的大学生心理健康咨询平台设计与实现 Spring Boot框架下大学生心理健康咨询管理系统开发

计算机毕业设计springboot大学生心理健康咨询系统jpmyh &#xff08;配套有源码 程序 mysql数据库 论文&#xff09; 本套源码可以在文本联xi,先看具体系统功能演示视频领取&#xff0c;可分享源码参考。随着互联网技术的飞速发展&#xff0c;大学生的心理健康问题逐渐受到广泛…

作者头像 李华