当前位置: 首页 > article >正文

滑动窗口详解:解决无重复字符的最长子串问题

滑动窗口详解:解决无重复字符的最长子串问题

在算法面试中,“无重复字符的最长子串”问题是一个经典题目,不仅考察基础数据结构的运用,还能够反映你的逻辑思维能力。而在解决这个问题时,滑动窗口(Sliding Window)算法可以说是绝对的明星。本篇文章将带你深入理解滑动窗口的原理,并通过代码和案例一步步解析如何应用它解决该问题。


问题描述

给定一个字符串 s,请你找出其中不含有重复字符的最长子串的长度。

例如:

  • 输入:s = "abcabcbb",输出:3,因为最长子串是 “abc”。
  • 输入:s = "bbbbb",输出:1,因为最长子串是 “b”。
  • 输入:s = "pwwkew",输出:3,因为最长子串是 “wke”。

乍一看,这个问题可能让人觉得需要两重循环暴力解决。但我们如何优化到线性时间复杂度呢?这时,滑动窗口大显身手。


滑动窗口的原理

滑动窗口是一种双指针技巧,通常用来解决子数组或子串相关问题。它的核心思想是:

  1. 用两个指针标记窗口的左右边界
  2. 动态调整窗口大小以满足问题条件
  3. 在移动窗口的过程中记录答案

滑动窗口的优势在于,它可以避免不必要的重复计算,从而优化时间复杂度。


滑动窗口解法详解

我们来看具体的实现:

# 主函数:寻找无重复字符的最长子串
def length_of_longest_substring(s: str) -> int:# 初始化变量char_set = set()  # 用于存储窗口内的字符left = 0          # 左指针max_length = 0    # 记录最大子串长度# 遍历字符串for right in range(len(s)):# 当字符重复时,缩小窗口while s[right] in char_set:print(f"重复字符:{s[right]},移除左侧字符:{s[left]}")char_set.remove(s[left])left += 1# 将当前字符加入窗口char_set.add(s[right])# 更新最大长度current_length = right - left + 1max_length = max(max_length, current_length)print(f"窗口:{s[left:right+1]},当前长度:{current_length}")return max_length

代码详解
  1. 初始化:

    • char_set 是一个集合,用来存储当前窗口中的字符。
    • left 是滑动窗口的左边界。
    • max_length 用来记录当前最长的无重复子串长度。
  2. 遍历字符串:

    • right 指针扩展窗口。
    • 如果 s[right]char_set 中,则表示出现了重复字符,需要通过移动左指针 left 来缩小窗口,直到重复字符被移除。
  3. 更新答案:

    • 每次调整窗口后,计算当前窗口的长度,并更新 max_length

示例运行

我们以 s = "abcabcbb" 为例,逐步运行代码:

  1. 初始状态:窗口为空,left = 0max_length = 0
  2. 遍历字符串:
    • 右指针移动到 0:窗口为 “a”,max_length = 1
    • 右指针移动到 1:窗口为 “ab”,max_length = 2
    • 右指针移动到 2:窗口为 “abc”,max_length = 3
    • 右指针移动到 3:字符 “a” 重复,移除左侧的 “a”,窗口为 “bca”。
    • ……
    • 最终输出 max_length = 3

时间复杂度分析
  • 时间复杂度: 每个字符最多被左指针和右指针访问一次,时间复杂度为 O(n)
  • 空间复杂度: 需要一个集合存储窗口内的字符,空间复杂度为 O(k),其中 k 是字符集的大小(对于英文字符,k 最多为 26)。

常见扩展问题
  1. 找出最长子串本身:
    如果不仅要返回长度,还要返回子串本身,可以在代码中记录窗口的起始位置。
def longest_substring(s: str) -> str:char_set = set()left = 0max_length = 0start = 0for right in range(len(s)):while s[right] in char_set:char_set.remove(s[left])left += 1char_set.add(s[right])if right - left + 1 > max_length:max_length = right - left + 1start = leftreturn s[start:start + max_length]
  1. 滑动窗口在其他场景中的应用:
    • 滑动窗口求和问题:固定窗口大小,求最大子数组和。
    • 双指针应用于字符串匹配问题,如查找最短覆盖子串。

总结

通过滑动窗口,我们可以优雅地解决“无重复字符的最长子串”问题。这种算法思想不仅高效,还能迁移到很多其他问题中。作为算法学习者,理解并掌握滑动窗口是进阶的必经之路。

如果你对滑动窗口有任何问题,或者想探索更多类似问题的解决方法,欢迎在评论区与我交流!

相关文章:

滑动窗口详解:解决无重复字符的最长子串问题

滑动窗口详解:解决无重复字符的最长子串问题 在算法面试中,“无重复字符的最长子串”问题是一个经典题目,不仅考察基础数据结构的运用,还能够反映你的逻辑思维能力。而在解决这个问题时,滑动窗口(Sliding …...

力扣-链表-19 删除链表倒数第N个节点

思路 链表题目中操作链表的需要找到要操作节点的上一个节点,所以cur是当前想要操作的节点上一个节点 代码 class Solution { public:ListNode* removeNthFromEnd(ListNode* head, int n) {ListNode* dummy_head new ListNode();dummy_head->next head;int s…...

算法题(49):反转链表II

审题: 需要我们对指定范围的链表进行反转,并返回反转后链表的头结点 思路: 方法一:vector法 我们先遍历一次链表,并把数据对应的存在数组中,然后利用数组的reverse方法进行反转数据,最后再遍历一…...

基于SpringBoot多数据源解决方案

最近在学习SpringBoot的时候,需要同时用两个不同的数据库连接服务,在网上学习了之后,下文以连接一个MySQL数据库和一个SqlServer数据库为例。 配置数据源连接信息 在配置文件中,配置对应的数据库连接信息,相比于单数…...

通过案例研究二项分布和泊松分布之间关系(2)

通过案例研究二项分布和泊松分布之间关系 2. 汽车出事故的概率p与保险公司盈利W之间的关系3.通过遗传算法多次迭代计算控制p为多少时公司盈利最大(1) 计算过程(2) 结果及分析(计算过程详见附录二程序) 4.改变思路求解固定p为0.01时,保险费用如何设置公司可获得最大利润(1)计算过…...

RISC-V读书笔记4

目录 乘法与除法 RV32F 和 RV32D:单精度和双精度浮点数 原子操作 压缩指令 向量 乘法与除法 RV32M属于扩展的指令,主要扩展的就是便捷的乘法和除法指令。 除法: 商 (被除数− 余数) 除数 被除数 除数 商 余数 余数 被除数− (商 …...

【Uniapp-Vue3】request各种不同类型的参数详解

一、参数携带 我们调用该接口的时候需要传入type参数。 第一种 路径名称?参数名1参数值1&参数名2参数值2 第二种 uni.request({ url:"请求路径", data:{ 参数名:参数值 } }) 二、请求方式 常用的有get,post和put 三种,默认是get请求。…...

大数据学习之SCALA分布式语言三

7.集合类 111.可变set一 112.可变set二 113.不可变MAP集合一 114.不可变MAP集合二 115.不可变MAP集合三 116.可变map一 package com . itbaizhan . chapter07 //TODO 2. 使用 mutable.Map 前导入如下包 import scala . collection . mutable // 可变 Map 集合 object Ma…...

深度解析:MyBatis-Plus实现分页查询的封装!

全文目录: 开篇语前言摘要概述什么是分页查询?为什么选择 MyBatis-Plus?本文目标 源码解析分页插件核心逻辑 使用案例分享1. 配置 MyBatis-Plus 分页插件2. 定义分页查询方法3. Controller 层调用 应用场景案例优缺点分析优点缺点 核心类方法…...

第05章 14 绘制人脸部的PolyData并使用小圆锥体来展现法线

在VTK中,绘制人脸部的PolyData并使用小圆锥体来展现法线是一个常见的任务。这个过程可以通过以下步骤实现: 读取人脸部的PolyData:可以使用VTK的读取模块读取一个包含人脸部的.vtk或.obj文件。计算法线:使用VTK的vtkPolyDataNorm…...

基于微信小程序的电子商城购物系统设计与实现(LW+源码+讲解)

专注于大学生项目实战开发,讲解,毕业答疑辅导,欢迎高校老师/同行前辈交流合作✌。 技术范围:SpringBoot、Vue、SSM、HLMT、小程序、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容:…...

2000-2020年各省第二产业增加值占GDP比重数据

2000-2020年各省第二产业增加值占GDP比重数据 1、时间:2000-2020年 2、来源:国家统计局、统计年鉴 3、指标:行政区划代码、地区名称、年份、第二产业增加值占GDP比重 4、范围:31省 5、指标解释:第二产业增加值占GDP比重…...

【Docker】Docker入门了解

文章目录 Docker 的核心概念Docker 常用命令示例:构建一个简单的 C 应用容器1. 创建 C 应用2. 创建 Dockerfile3. 构建镜像4. 运行容器 Docker 优势学习 Docker 的下一步 **一、Docker 是什么?****为什么 C 开发者需要 Docker?** **二、核心概…...

【C语言分支与循环结构详解】

目录 ---------------------------------------begin--------------------------------------- 一、分支结构 1. if语句 2. switch语句 二、循环结构 1. for循环 2. while循环 3. do-while循环 三、嵌套结构 结语 -----------------------------------------end----…...

java求职学习day18

常用的设计原则和设计模式 1 常用的设计原则(记住) 1.1 软件开发的流程 需求分析文档、概要设计文档、详细设计文档、编码和测试、安装和调试、维护和升级 1.2 常用的设计原则 (1)开闭原则(Open Close Principle…...

初阶2 类与对象

本章重点 上篇1.面向过程和面向对象初步认识2.类的引入---结构体3.类的定义3.1 语法3.2 组成3.3 定义类的两种方法: 4.类的访问限定符及封装4.1 访问限定符4.2封装---面向对象的三大特性之一 5.类的作用域6.类的实例化7.类对象模型7.1 如何计算类对象的大小 8.this指…...

蓝桥杯模拟算法:多项式输出

P1067 [NOIP2009 普及组] 多项式输出 - 洛谷 | 计算机科学教育新生态 这道题是一道模拟题&#xff0c;我们需要分情况讨论&#xff0c;我们需要做一下分类讨论 #include <iostream> #include <cstdlib> using namespace std;int main() {int n;cin >> n;for…...

Python的那些事第三篇:Python编程的“调味料”与“交流术”运算符与输入输出

运算符与输入输出&#xff1a;Python编程的“调味料”与“交流术” 在编程的世界里&#xff0c;Python不仅仅是一门语言&#xff0c;它更像是一位充满智慧的厨师&#xff0c;而运算符和输入输出则是它手中的“调味料”和“交流术”。没有这些工具&#xff0c;代码就会像没有加…...

如何利用AI工具来进行数据分析

利用AI工具进行数据分析可以显著提高效率和准确性&#xff0c;以下是详细步骤和方法&#xff1a; 1. 明确分析目标 在开始数据分析之前&#xff0c;首先需要明确分析的目标和问题。这包括确定需要解决的问题、期望的见解或结果&#xff0c;以及选择合适的AI工具和方法。 2. …...

深度剖析C++17中的std::optional:处理可能缺失值的利器

文章目录 一、基本概念与设计理念二、构建与初始化&#xff08;一&#xff09;默认构造&#xff08;二&#xff09;值初始化&#xff08;三&#xff09;使用std::make_optional&#xff08;四&#xff09;使用std::nullopt 三、访问值&#xff08;一&#xff09;value()&#x…...

MySQL用户授权、收回权限与查看权限

【图书推荐】《MySQL 9从入门到性能优化&#xff08;视频教学版&#xff09;》-CSDN博客 《MySQL 9从入门到性能优化&#xff08;视频教学版&#xff09;&#xff08;数据库技术丛书&#xff09;》(王英英)【摘要 书评 试读】- 京东图书 (jd.com) MySQL9数据库技术_夏天又到了…...

每日一题 429. N 叉树的层序遍历

429. N 叉树的层序遍历 /*class Solution { public:vector<vector<int>> levelOrder(Node* root) {queue<Node*> que;que.push(root);vector<vector<int>> ans;if(root nullptr){return ans;}while(!que.empty()){int sizeQue que.size();vec…...

【Maui】注销用户,采用“手势”点击label弹窗选择

文章目录 前言一、问题描述二、解决方案三、软件开发&#xff08;源码&#xff09;3.1 方法一&#xff1a;前端绑定3.2 方法二&#xff1a;后端绑定3.3 注销用户的方法 四、项目展示 前言 .NET 多平台应用 UI (.NET MAUI) 是一个跨平台框架&#xff0c;用于使用 C# 和 XAML 创…...

如何将xps文件转换为txt文件?xps转为pdf,pdf转为txt,提取pdf表格并转为txt

文章目录 xps转txt方法一方法二 pdf转txt整页转txt提取pdf表格&#xff0c;并转为txt 总结另外参考XPS文件转换为TXT文件XPS文件转换为PDF文件PDF文件转换为TXT文件提取PDF表格并转为TXT示例代码&#xff08;部分&#xff09; 本文测试代码已上传&#xff0c;路径如下&#xff…...

Object类(2)

大家好&#xff0c;今天我们继续来看看Object类中一些成员方法&#xff0c;这些方法在实际中有很大的用处&#xff0c;话不多说&#xff0c;来看。 注&#xff1a;所有类都默认继承Object类的&#xff0c;所以可调用Object类中的方法&#xff0c;如equals&#xff0c;也可以发生…...

BGP分解实验·11——路由聚合与条件性通告(3)

续接上&#xff08;2&#xff09;的实验。其拓扑如下&#xff1a; 路由聚合的负向也就是拆分&#xff0c;在有双出口的情况下&#xff0c;在多出口做流量分担是优选方法之一。 BGP可以根据指定来源而聚合路由&#xff0c;在产生该聚合路由的范围内的条目注入到本地BGP表后再向…...

无用的知识又增加了:is_assignable means?

std::pair的默认operator被delete掉了&#xff0c;取而代之的是两个enable_if版本。 为什么这么设计&#xff0c;我的理解是在std::map里&#xff0c;已经保存的元素的key值是不能被修改的&#xff0c;比如 注意&#xff0c;下面的代码会修改key值&#xff0c;编译时出现错误…...

MOS的体二极管能通多大电流

第一个问题&#xff1a;MOS导通之后电流方向可以使任意的&#xff0c;既可以从D到S&#xff0c;也可以从S到D。 第二个问题&#xff1a;MOS里面的体二极管电流可以达到几百安培&#xff0c;这也就解释了MOS选型的时候很少考虑体二极管的最大电流&#xff0c;而是考虑DS之间电流…...

C语言【基础篇】之流程控制——掌握三大结构的奥秘

流程控制 &#x1f680;前言&#x1f99c;顺序结构&#x1f4af; 定义&#x1f4af;执行规则 &#x1f31f;选择结构&#x1f4af;if语句&#x1f4af;switch语句&#x1f4af;case穿透规则 &#x1f914;循环结构&#x1f4af;for循环&#x1f4af;while循环&#x1f4af;do -…...

Node.js下载安装及环境配置教程 (详细版)

Node.js&#xff1a;是一个基于 Chrome V8 引擎的 JavaScript 运行时&#xff0c;用于构建可扩展的网络应用程序。Node.js 使用事件驱动、非阻塞 I/O 模型&#xff0c;使其非常适合构建实时应用程序。 Node.js 提供了一种轻量、高效、可扩展的方式来构建网络应用程序&#xff0…...