滑动窗口解最长不重复子串时间复杂度
时间:2025-08-29 14:57:37 364浏览 收藏
本文深入探讨了使用滑动窗口算法求解字符串最长无重复子串长度的问题,并针对常见实现方式中存在的时间复杂度问题进行了分析。最初的方案虽然使用了滑动窗口,但由于内部循环的存在,导致时间复杂度并非严格的O(n)。为了优化算法性能,本文提出了一种基于Map数据结构的改进方案,将时间复杂度降低到O(n)。通过代码示例和详细的步骤解释,本文旨在帮助读者理解滑动窗口算法的原理,并掌握优化技巧,从而高效解决字符串处理中的相关问题。该优化后的算法在处理大规模字符串时具有显著优势。
本文旨在深入解析求解字符串中最长无重复子串长度的滑动窗口算法。我们将分析一种常见的实现方式,指出其潜在的时间复杂度问题,并提供一种更优的、时间复杂度为 O(n) 的解决方案。通过代码示例和详细解释,帮助读者理解算法原理并掌握优化技巧。
问题描述
给定一个字符串,找出其中最长且不包含重复字符的子串的长度。例如:
- 输入 "abcabcbb",答案是 3 (对应子串 "abc")
- 输入 "bbbbb",答案是 1 (对应子串 "b")
- 输入 "pwwkew",答案是 3 (对应子串 "wke")
初始方案分析
最初的解决方案采用滑动窗口的思想,使用一个对象 (storage.cache) 来缓存字符及其索引。虽然看起来像是滑动窗口,但由于在遇到重复字符时,存在一个内部循环,导致其时间复杂度并非严格的 O(n)。
以下是原始代码:
var lengthOfLongestSubstring = function(str) { // Create storage object for caching let storage = { longestSubStringLength: 0, longestSubString: 0, cache: { subString: '' } }; // Loop through string for (let i = 0; i < str.length; i++) { let char = str[i]; if (!storage.cache[char] && storage.cache[char] !== 0) { // If current letter is not in storage, add it and extend current substring storage.cache[char] = i; storage.cache.subString += char; } else { // If current letter is already in storage, start a new round let previousCache = storage.cache; storage.cache = { subString: '' }; if (previousCache[char] + 1 !== i) { // If there are letters in-between storage.cache.subString = str.substring(previousCache[char] + 1, i); for (let j = previousCache[char]; j < i; j++) { storage.cache[str[j]] = j; } } storage.cache[char] = i; storage.cache.subString += char; } // If current substring is the longest, update it in storage if (storage.cache.subString.length > storage.longestSubStringLength) { storage.longestSubStringLength = storage.cache.subString.length; storage.longestSubString = storage.cache.subString; } } return storage.longestSubStringLength; };
问题在于 else 分支中的内部 for 循环:
for (let j = previousCache[char]; j < i; j++) { storage.cache[str[j]] = j; }
这个循环在遇到重复字符时,会迭代更新 storage.cache 中位于重复字符之间的字符的索引。在最坏情况下,例如 "abcdefghabcdefgh",这个内部循环可能会执行多次,导致整体时间复杂度高于 O(n)。 更准确地说,其时间复杂度接近 O(n*m),其中 m 是最长不重复子串的平均长度。
优化方案:O(n) 时间复杂度的滑动窗口
为了实现真正的 O(n) 时间复杂度,我们可以使用 Map 数据结构来存储字符及其索引。Map 提供了快速的查找和更新操作。
以下是优化后的代码:
const lengthOfLongestSubstring = str => { let cnt = 0; let n = str.length; let answer = 0; let map = new Map(); // to store the strings and their length for (let start = 0, end = 0; end < n; end++) { // slide // move start if the character is already in the map if (map.has(str[end])) start = Math.max(map.get(str[end]), start); answer = Math.max(answer, end - start + 1); // longest string map.set(str[end], end + 1); cnt++ } return [str, `lookups: ${cnt} lookups:`, "answer", answer]; } ["abcabcbb", "bbbbb", "pwwkew", "abcdefghabcdefgh"].forEach(str => console.log(lengthOfLongestSubstring(str).join(" ")))
代码解释:
- map: 使用 Map 来存储字符及其在字符串中的下一个位置(索引 + 1)。
- start 和 end: start 指向当前无重复子串的起始位置,end 指向当前遍历的字符。
- 滑动窗口: end 指针不断向右移动,扩展窗口。
- 重复字符处理: 如果 map 中已经存在当前字符 str[end],则将 start 指针移动到 map.get(str[end]) 和当前 start 的较大值处。 这是关键步骤,确保 start 始终指向当前无重复子串的有效起始位置。Math.max 的使用是为了避免 start 指针回退,这种情况可能发生在字符串中字符重复出现多次,且重复字符的索引小于当前的 start 值。
- 更新最大长度: 每次迭代都更新 answer,即最长无重复子串的长度。
- 更新 map: 将当前字符 str[end] 及其下一个位置 end + 1 存入 map。
时间复杂度分析:
- 外层循环 for (let start = 0, end = 0; end < n; end++) 遍历字符串一次,O(n)。
- Map 的 has、get 和 set 操作的平均时间复杂度为 O(1)。
因此,整体时间复杂度为 O(n)。
空间复杂度分析:
空间复杂度为 O(min(m, n)),其中 m 是字符集的大小,n 是字符串的长度。这是因为 Map 最多存储 m 个不同的字符及其索引。
总结
通过使用 Map 数据结构和滑动窗口技术,我们可以高效地解决最长无重复子串问题,并将时间复杂度优化到 O(n)。 关键在于正确地维护滑动窗口的起始位置,并利用 Map 快速查找和更新字符的索引。 这种方法不仅提高了效率,还使代码更简洁易懂。
文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《滑动窗口解最长不重复子串时间复杂度》文章吧,也可关注golang学习网公众号了解相关技术文章。
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
426 收藏
-
341 收藏
-
105 收藏
-
400 收藏
-
111 收藏
-
359 收藏
-
143 收藏
-
129 收藏
-
471 收藏
-
460 收藏
-
150 收藏
-
399 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 511次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 498次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习