leetcode 面试经典 150 题:无重复字符的最长子串
| 链接 | 无重复字符的最长子串 |
|---|---|
| 题序号 | 3 |
| 类型 | 字符串 |
| 解题方法 | 滑动窗口 |
| 难度 | 中等 |
题目
给定一个字符串 s ,请你找出其中不含有重复字符的 最长子串 的长度。
示例 1:
输入: s = “abcabcbb”
输出: 3
解释: 因为无重复字符的最长子串是 “abc”,所以其长度为 3。示例 2:
输入: s = “bbbbb”
输出: 1
解释: 因为无重复字符的最长子串是 “b”,所以其长度为 1。示例 3:
输入: s = “pwwkew”
输出: 3
解释: 因为无重复字符的最长子串是 “wke”,所以其长度为 3。请注意,你的答案必须是 子串 的长度,“pwke” 是一个子序列,不是子串。提示:
0 <= s.length <= 5 * 104
s 由英文字母、数字、符号和空格组成
解题
-
核心要点:这个问题可以通过滑动窗口的方法来解决。滑动窗口表示当前考虑的子串的范围,我们通过移动窗口的边界来找到不含有重复字符的最长子串。
- 初始化:定义两个指针 left 和 right 分别表示窗口的左右边界,以及一个哈希表 char_index_map 来存储字符及其索引。
- 扩展窗口:将 right 指针向右移动,扩展窗口,直到遇到重复的字符。
- 更新最大长度:在每次移动 right 指针时,更新不含有重复字符的子串的最大长度。
- 收缩窗口:当遇到重复的字符时,移动 left 指针,收缩窗口,直到重复的字符被移出窗口。
- 重复步骤2-4:直到 right 指针到达字符串的末尾。
-
时间复杂度: O(n)
-
空间复杂度:O(n)
-
c++ 实现算法:
class Solution {
public:int lengthOfLongestSubstring(string s) {unordered_map<char, int> map; // 哈希表记录字符的最新索引,map 的键是字符,值是该字符的最后出现的索引位置int max_len = 0; // 最长子串长度int left = 0; // 滑动窗口的左边界for (int right = 0; right < s.size(); ++right) {// 如果当前字符在窗口内出现过,更新窗口的左边界//如果 map.find(s[right]) 返回的不是 map.end(),意味着哈希表中存在字符 s[right],//也就是说,字符 s[right] 在哈希表中出现过。//如果返回 map.end(),则说明哈希表中没有这个字符,即字符 s[right] 没有在哈希表中出现过。//map[s[right]] >= left,字符 s[right] 在窗口中的 上一次出现的位置 //大于等于当前窗口的左边界 left,那么就说明它是重复的。if (map.find(s[right]) != map.end() && map[s[right]] >= left) {//将 left 更新为字符 s[right] 最新出现位置的下一个位置left = map[s[right]] + 1;}// 更新当前字符的最新位置map[s[right]] = right;// 计算当前窗口的大小,更新最大长度max_len = max(max_len, right - left + 1);}return max_len;}
};
- 演示:以示例1为例
遍历字符串 s,right 从 0 到 7:
right = 0,字符 ‘a’: map.find(s[right]) != map.end() 检查字符 ‘a’ 是否已经出现过。由于 map 为空,所以字符 ‘a’ 不在其中。 更新哈希表:map[‘a’] = 0。字符 ‘a’ 的最新索引是 0。计算当前窗口大小:right - left + 1 = 0 - 0 + 1 = 1。 更新 max_len:max_len = max(0,1) = 1。
right = 1,字符 ‘b’: map.find(s[right]) != map.end() 检查字符 ‘b’ 是否已经出现过。字符 ‘b’ 不在哈希表中。 更新哈希表:map[‘b’] = 1。字符 ‘b’ 的最新索引是 1。计算当前窗口大小:right - left + 1 = 1 - 0 + 1 = 2。 更新 max_len:max_len = max(1, 2) = 2。
right = 2,字符 ‘c’: map.find(s[right]) != map.end() 检查字符 ‘c’ 是否已经出现过。字符 ‘c’ 不在哈希表中。 更新哈希表:map[‘c’] = 2。字符 ‘c’ 的最新索引是 2。计算当前窗口大小:right - left + 1 = 2 - 0 + 1 = 3。 更新 max_len:max_len = max(2, 3) = 3。
right = 3,字符 ‘a’: map.find(s[right]) != map.end() 检查字符 ‘a’ 是否已经出现过。字符 ‘a’ 已经在 map 中,且它的索引是 0(小于 right = 3)。 由于 ‘a’ 重复了,我们需要更新滑动窗口的左边界: left = map[‘a’] + 1 = 0 + 1 = 1,新的左边界是 1,即跳过 a 在左边的位置。 更新哈希表:map[‘a’] = 3。字符 ‘a’ 的最新索引是 3。 计算当前窗口大小:right - left + 1 = 3 - 1 + 1 = 3。 max_len 仍然是 3(不更新)。
right = 4,字符 ‘b’: map.find(s[right]) != map.end() 检查字符 ‘b’ 是否已经出现过。字符 ‘b’ 已经在 map 中,且它的索引是 1(小于 right = 4)。 由于 ‘b’ 重复了,我们需要更新滑动窗口的左边界: left = map[‘b’] + 1 = 1 + 1 = 2,新的左边界是 2,即跳过 b 在左边的位置。 更新哈希表:map[‘b’] = 4。字符 ‘b’ 的最新索引是 4。 计算当前窗口大小:right - left + 1 = 4 - 2 + 1 = 3。 max_len 仍然是 3(不更新)。
right = 5,字符 ‘c’: map.find(s[right]) != map.end() 检查字符 ‘c’ 是否已经出现过。字符 ‘c’ 已经在 map 中,且它的索引是 2(小于 right = 5)。 由于 ‘c’ 重复了,我们需要更新滑动窗口的左边界: left = map[‘c’] + 1 = 2 + 1 = 3,新的左边界是 3,即跳过 c 在左边的位置。 更新哈希表:map[‘c’] = 5。字符 ‘c’ 的最新索引是 5。 计算当前窗口大小:right - left + 1 = 5 - 3 + 1 = 3。 max_len 仍然是 3(不更新)。
right = 6,字符 ‘b’: map.find(s[right]) != map.end() 检查字符 ‘b’ 是否已经出现过。字符 ‘b’ 已经在 map 中,且它的索引是 4(小于 right = 6)。 由于 ‘b’ 重复了,我们需要更新滑动窗口的左边界: left = map[‘b’] + 1 = 4 + 1 = 5,新的左边界是 5,即跳过 b在左边的位置。 更新哈希表:map[‘b’] = 6。字符 ‘b’ 的最新索引是 6。 计算当前窗口大小:right - left + 1 = 6 - 5 + 1 = 2。 max_len 仍然是 3(不更新)。
right = 7,字符 ‘b’: map.find(s[right]) != map.end() 检查字符 ‘b’ 是否已经出现过。字符 ‘b’ 已经在 map 中,且它的索引是 6(小于 right = 7)。 由于 'b’重复了,我们需要更新滑动窗口的左边界: left = map[‘b’] + 1 = 6 + 1 = 7,新的左边界是 7,即跳过 b 在左边的位置。 更新哈希表:map[‘b’] = 7。字符 ‘b’ 的最新索引是 7。 计算当前窗口大小:right - left + 1 = 7 - 7 + 1 = 1。 max_len 仍然是 3(不更新)。
- c++ demo:
#include <iostream>
#include <unordered_map>
#include <algorithm>
using namespace std;class Solution {
public:int lengthOfLongestSubstring(string s) {unordered_map<char, int> map; // 哈希表记录字符的最新索引,map 的键是字符,值是该字符的最后出现的索引位置int max_len = 0; // 最长子串长度int left = 0; // 滑动窗口的左边界for (int right = 0; right < s.size(); ++right) {// 如果当前字符在窗口内出现过,更新窗口的左边界//如果 map.find(s[right]) 返回的不是 map.end(),意味着哈希表中存在字符 s[right],//也就是说,字符 s[right] 在哈希表中出现过。//如果返回 map.end(),则说明哈希表中没有这个字符,即字符 s[right] 没有在哈希表中出现过。//map[s[right]] >= left,字符 s[right] 在窗口中的 上一次出现的位置 //大于等于当前窗口的左边界 left,那么就说明它是重复的。if (map.find(s[right]) != map.end() && map[s[right]] >= left) {//将 left 更新为字符 s[right] 最新出现位置的下一个位置left = map[s[right]] + 1;}// 更新当前字符的最新位置map[s[right]] = right;// 计算当前窗口的大小,更新最大长度max_len = max(max_len, right - left + 1);}return max_len;}
};int main() {Solution sol;string s = "abcabcbb";cout << "The length of the longest substring without repeating characters is: " << sol.lengthOfLongestSubstring(s) << endl;return 0;
}
相关文章:
leetcode 面试经典 150 题:无重复字符的最长子串
链接无重复字符的最长子串题序号3类型字符串解题方法滑动窗口难度中等 题目 给定一个字符串 s ,请你找出其中不含有重复字符的 最长子串 的长度。 示例 1: 输入: s “abcabcbb” 输出: 3 解释: 因为无重复字符的最长子串是 “abc”,所以其长度为 3。 …...
0101多级nginx代理websocket配置-nginx-web服务器
1. 前言 项目一些信息需要通过站内信主动推动给用户,使用websocket。web服务器选用nginx,但是域名是以前通过阿里云申请的,解析ip也是阿里云的服务器,甲方不希望更换域名。新的系统需要部署在内网服务器,简单拓扑图如…...
【前端】Jquery拍照,通过PHP将base64编码数据转换成PNG格式,并保存图像到本地
目录 一、需求 二、开发语言 三、效果 四、业务逻辑: 五、web端调用摄像头 六、示例代码 1、前端 2、后端 一、需求 web端使用jquery调用摄像头拍照,并使用PHP把base64编码转换成png格式图片,下载到本地。 由于js不能指定图片存储的…...
websocket再项目中的使用
WebSocket在项目中的使用主要包括以下几个方面: WebSocket的基本概念和原理: 定义:WebSocket是一种基于TCP的协议,实现了浏览器与服务器之间的全双工通信。它通过HTTP/1.1协议的101状态码进行握手,建立连接…...
ajax同步执行async:false无效的解决方法
无效的情况: function ManHourCheck() {var StartDate $("#StartDate").val();//日报日期var EndDate $("#EndDate").val();//完成日期var UserID $("#UserID").val();//员工ID$.ajax({async: false,//加了这一行也没用!!!!!!!!!!…...
基于Qt的登陆界面设计
目标 自由发挥登录界面的应用场景,实现一个登录窗口的界面。 要求:每行代码都要有注释 代码 // 设置窗口大小为600x400像素 this->resize(600,400); // 设置窗口标题为"TheWitcher 巫师3:狂猎" this->setWindowTitle(&qu…...
HarmonyOS 输入框组件:TextInput 和 TextArea 深度解析
输入框组件是移动端开发中最常见的组件之一,常用于响应用户的输入操作,比如评论区的文本输入、聊天框的消息输入、表单内容填写等场景。在 HarmonyOS 中,TextInput 和 TextArea 分别用于单行和多行输入操作。除此之外,它们还可以与…...
【Golang】 Go 语言中的 Struct、JSON 和 Map 互转:详细指南
Go 语言中的 Struct、JSON 和 Map 互转:详细指南 在 Go 语言中,处理 JSON 数据、结构体类型和映射(map)是与 API、配置或数据库交互时非常常见的任务。理解如何在这些数据类型之间无缝转换对于高效的 Go 编程至关重要。以下是如何将 Go 结构体转换为 JSON、将 JSON 转换为…...
Azure Function流式返回
最近用azure function做了一个api和llm交互,需要流式返回。但是默认不支持流返回,搜索了一下。记录。 官方文档:https://techcommunity.microsoft.com/blog/azurecompute/azure-functions-support-for-http-streams-in-python-is-now-in-prev…...
智能座舱进阶-应用框架层-Jetpack主要组件
Jetpack的分类 1. DataBinding:以声明方式将可观察数据绑定到界面元素,通常和ViewModel配合使用。 2. Lifecycle:用于管理Activity和Fragment的生命周期,可帮助开发者生成更易于维护的轻量级代码。 3. LiveData: 在底层数据库更…...
GitLab分支管理策略和最佳实践
分支管理是 Git 和 GitLab 中非常重要的部分,合理的分支管理可以帮助团队更高效地协作和开发。以下是一些细化的分支管理策略和最佳实践: 1. 分支命名规范 • 主分支:通常命名为 main 或 master,用于存放稳定版本的代码。 • …...
【Unity】【VR开发】实现VR屏幕共享应用的几个重要插件和参考资料分享
【背景】 做了一个可以在局域网远程屏幕的VR应用,如果有相同兴趣的朋友也可以参考下我用的几个插件。 【使用或相关的关键插件】 piping server:这个是最基底的插件,基于它实现的信令通信。 https://github.com/nwtgck/piping-server/blob…...
数据结构---------二叉树前序遍历中序遍历后序遍历
以下是用C语言实现二叉树的前序遍历、中序遍历和后序遍历的代码示例,包括递归和非递归(借助栈实现)两种方式: 1. 二叉树节点结构体定义 #include <stdio.h> #include <stdlib.h>// 二叉树节点结构体 typedef struct…...
浏览器引入elasticsearch-head插件
elasticsearch-head插件下载: 链接: https://pan.baidu.com/s/1Dz3aU42HZCNg45iJoDOsMg?pwduvhg 提取码: uvhg 1、打开浏览器设置 2、选择拓展程序 3、选择elasticsearch-head插件下载 4、打开es-head插件 5、修改ip 6、登录...
【ELK】Filebeat采集Docker容器日志
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 介绍filebeat是如何工作的 使用部署filebeat 介绍 Filebeat 是一个用于转发和集中日志数据的轻量级传送器。 Filebeat 作为agent安装在服务器上,监视指…...
异步线程池与CountDownLatch
异步线程池 顾名思义,一个专门用来处理异步任务的线程池。可以避免线程的开销以及非阻塞的执行任务。 CountDownLatch 一个同步工具类,用于 让一个或多个线程等待一组操作完成。 业务场景 支付订单时,用户可以使用多张优惠劵,…...
在图像上显示掩码、框和点的通用函数
在图像上显示掩码、框和点的通用函数 背景介绍函数实现与用途1. 显示掩码函数:`show_mask`2. 显示边界框函数:`show_box`3. 在图像上显示点函数:`show_points`4. 综合显示框和点函数:`show_points_and_boxes_on_image`5. 显示掩码并返回图像函数:`show_mask_on_image`6. 显…...
基于Matlab的变压器仿真模型建模方法(11):三相三绕组换流变压器的建模仿真
1.概述 换流变压器是直流输电系统中的关键设备,主要负责连接交流和直流系统,并实现电能的转换与传输。换流变压器在直流输电系统中的主要用途包括:传送电力:将电能从交流系统传输到直流系统或从直流系统传输到交流系统;电压变换:把交流系统电压变换到换流器所需的换相电压…...
代码随想录算法训练营day46|动态规划part12
今天就结束动态规划章节了,以后还要多加练习。 今天的两道题都很有难度,647回文子串的思路非常巧妙,因为用一维dp数组比较难表示子串的起点和终点,所以需要用二维dp数组表示,dp[i][j]表示以i为起点,j为终点…...
【C语言】头文件
所有学习过C语言的朋友都熟悉这样一段代码: #include <stdio.h>int main(int argc, char *argv[]) {return 0; }那么,你真的了解 <stdio.h> 吗? <stdio…...
Python胶水代码变高性能引擎(Mojo原生编译实战手记)
第一章:Python胶水代码变高性能引擎(Mojo原生编译实战手记)Python 以其简洁语法和丰富生态成为数据科学与系统集成的“胶水语言”,但其解释执行机制常在数值计算、实时推理等场景遭遇性能瓶颈。Mojo 作为新兴的系统级编程语言&…...
text2vec-base-chinese终极指南:如何用768维向量彻底改变中文语义理解
text2vec-base-chinese终极指南:如何用768维向量彻底改变中文语义理解 【免费下载链接】text2vec-base-chinese 项目地址: https://ai.gitcode.com/hf_mirrors/ai-gitcode/text2vec-base-chinese 还在为中文文本的语义匹配而头疼吗?传统的基于关…...
485总线硬件设计必看:电平匹配、TVS防护,还有exmodbus库快速上手
RS485是工业物联网的标配通信接口。合宙Air780EHV系列Cat.1模组凭借强大外设扩展能力(LCD、摄像头、以太网、CAN等)和LuatOS高效开发环境,支持TCP/MQTT/HTTP/Modbus等主流协议,是工业场景的高性价比之选。 本文聚焦RS485实战&…...
从SolidWorks到ROS:六自由度机械臂URDF模型转换实战指南
1. 从SolidWorks到ROS的桥梁:URDF模型转换概述 当你费尽心思在SolidWorks中完成了六自由度机械臂的三维建模,看着那些精密的齿轮和连杆在软件中流畅转动时,脑海中可能已经浮现出它在ROS环境中大展身手的场景。但问题来了:如何让这…...
Scratch飞翔小鸟游戏制作教程:从零开始打造你的第一个像素风小游戏
Scratch飞翔小鸟游戏制作教程:从零开始打造你的第一个像素风小游戏 当孩子们第一次接触编程时,往往会被复杂的代码和抽象的概念吓退。而Scratch就像一扇通往创意世界的大门,用积木式的编程方式让游戏开发变得触手可及。今天,我们将…...
clusterProfiler进阶指南:如何利用R语言进行多组学数据的功能富集分析与可视化
clusterProfiler进阶指南:如何利用R语言进行多组学数据的功能富集分析与可视化 在生物信息学领域,功能富集分析是将高通量组学数据转化为生物学洞见的关键步骤。作为R/Bioconductor生态中的明星工具,clusterProfiler以其强大的分析能力和丰富…...
PyTorch 2.8深度学习镜像入门必看:RTX 4090D环境验证与快速上手步骤
PyTorch 2.8深度学习镜像入门必看:RTX 4090D环境验证与快速上手步骤 1. 镜像概述与核心优势 PyTorch 2.8深度学习镜像专为RTX 4090D显卡优化设计,提供开箱即用的深度学习开发环境。这个镜像最显著的特点是免去了复杂的环境配置过程,让开发者…...
Qwen3.5-9B-AWQ-4bit惊艳效果:多对象复杂场景图中主次关系与逻辑推断展示
Qwen3.5-9B-AWQ-4bit惊艳效果:多对象复杂场景图中主次关系与逻辑推断展示 1. 模型能力概览 千问3.5-9B-AWQ-4bit是一款突破性的多模态AI模型,它能够像人类一样"看懂"图片并做出智能分析。不同于传统图像识别工具,这个模型最令人惊…...
AI报告文档审核赋能人才培养:IACheck打造环境检测人机协同审核虚拟仿真新体系
在环境检测行业持续走向精细化与规范化的过程中,报告审核能力逐渐成为影响整体质量的重要因素。然而,与检测设备和分析技术不断升级相比,审核人员的培养却长期依赖经验积累与“师带徒”模式,这种方式虽然能够传递实践经验…...
零成本玩转千问大模型!OpenClaw 配置秘籍(附每日500 Token福利)
🚀零成本玩转千问大模型!OpenClaw 配置秘籍(附每日500 Token福利) 这份教程将带你完成从环境准备到成功验证的全过程,让你手中的工具“活”起来。 前期准备:搭建基石 在开始代码操作之前,我们需…...
