当前位置:首页 > 文章列表 > 文章 > 前端 > 滑动窗口解最长不重复子串时间复杂度

滑动窗口解最长不重复子串时间复杂度

2025-08-29 14:57:37 0浏览 收藏

本文深入探讨了使用滑动窗口算法求解字符串最长无重复子串长度的问题,并针对常见实现方式中存在的时间复杂度问题进行了分析。最初的方案虽然使用了滑动窗口,但由于内部循环的存在,导致时间复杂度并非严格的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(" ")))

代码解释:

  1. map: 使用 Map 来存储字符及其在字符串中的下一个位置(索引 + 1)。
  2. start 和 end: start 指向当前无重复子串的起始位置,end 指向当前遍历的字符。
  3. 滑动窗口: end 指针不断向右移动,扩展窗口。
  4. 重复字符处理: 如果 map 中已经存在当前字符 str[end],则将 start 指针移动到 map.get(str[end]) 和当前 start 的较大值处。 这是关键步骤,确保 start 始终指向当前无重复子串的有效起始位置。Math.max 的使用是为了避免 start 指针回退,这种情况可能发生在字符串中字符重复出现多次,且重复字符的索引小于当前的 start 值。
  5. 更新最大长度: 每次迭代都更新 answer,即最长无重复子串的长度。
  6. 更新 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学习网公众号了解相关技术文章。

Java实现MinIO分片上传方法详解Java实现MinIO分片上传方法详解
上一篇
Java实现MinIO分片上传方法详解
Python递归深度设置技巧详解
下一篇
Python递归深度设置技巧详解
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    511次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    498次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 千音漫语:智能声音创作助手,AI配音、音视频翻译一站搞定!
    千音漫语
    千音漫语,北京熠声科技倾力打造的智能声音创作助手,提供AI配音、音视频翻译、语音识别、声音克隆等强大功能,助力有声书制作、视频创作、教育培训等领域,官网:https://qianyin123.com
    420次使用
  • MiniWork:智能高效AI工具平台,一站式工作学习效率解决方案
    MiniWork
    MiniWork是一款智能高效的AI工具平台,专为提升工作与学习效率而设计。整合文本处理、图像生成、营销策划及运营管理等多元AI工具,提供精准智能解决方案,让复杂工作简单高效。
    419次使用
  • NoCode (nocode.cn):零代码构建应用、网站、管理系统,降低开发门槛
    NoCode
    NoCode (nocode.cn)是领先的无代码开发平台,通过拖放、AI对话等简单操作,助您快速创建各类应用、网站与管理系统。无需编程知识,轻松实现个人生活、商业经营、企业管理多场景需求,大幅降低开发门槛,高效低成本。
    415次使用
  • 达医智影:阿里巴巴达摩院医疗AI影像早筛平台,CT一扫多筛癌症急慢病
    达医智影
    达医智影,阿里巴巴达摩院医疗AI创新力作。全球率先利用平扫CT实现“一扫多筛”,仅一次CT扫描即可高效识别多种癌症、急症及慢病,为疾病早期发现提供智能、精准的AI影像早筛解决方案。
    431次使用
  • 智慧芽Eureka:更懂技术创新的AI Agent平台,助力研发效率飞跃
    智慧芽Eureka
    智慧芽Eureka,专为技术创新打造的AI Agent平台。深度理解专利、研发、生物医药、材料、科创等复杂场景,通过专家级AI Agent精准执行任务,智能化工作流解放70%生产力,让您专注核心创新。
    449次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码