LeetCode 第3题:无重复字符的最长子串

题目描述

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

难度

中等

题目链接

https://leetcode.cn/problems/longest-substring-without-repeating-characters/

示例

示例 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 由英文字母、数字、符号和空格组成

解题思路

方法一:滑动窗口

这道题可以使用滑动窗口的方法解决。滑动窗口是一种在数组或字符串上使用双指针的技巧,通过移动左右指针形成一个窗口,在窗口内进行操作。

关键点:

  1. 使用两个指针(left和right)维护一个滑动窗口
  2. 使用HashSet记录窗口内的字符,保证无重复
  3. 当遇到重复字符时,移动左指针直到删除重复字符
  4. 在移动过程中更新最大长度

具体步骤:

  1. 初始化左指针left = 0,右指针right = 0
  2. 创建HashSet存储窗口内的字符
  3. 移动右指针,将字符加入Set:
    • 如果字符不在Set中,加入Set并更新最大长度
    • 如果字符在Set中,移动左指针并删除字符,直到没有重复
  4. 重复步骤3直到右指针到达字符串末尾

时间复杂度:O(n),其中n是字符串长度

空间复杂度:O(min(m,n)),其中m是字符集大小

方法二:优化的滑动窗口

使用Dictionary/HashMap代替HashSet,存储字符最后出现的位置,可以直接跳转左指针位置,避免一个个删除字符。

代码实现

C# 实现(基本滑动窗口)

public class Solution {
public int LengthOfLongestSubstring(string s) {
// 特殊情况处理
if (string.IsNullOrEmpty(s)) return 0; // 创建HashSet存储当前窗口中的字符
HashSet<char> window = new HashSet<char>();
int maxLength = 0;
int left = 0; // 右指针遍历字符串
for (int right = 0; right < s.Length; right++) {
// 如果发现重复字符,移动左指针直到删除重复字符
while (window.Contains(s[right])) {
window.Remove(s[left]);
left++;
} // 将当前字符加入窗口
window.Add(s[right]); // 更新最大长度
maxLength = Math.Max(maxLength, right - left + 1);
} return maxLength;
}
}

C# 实现(优化的滑动窗口)

public class Solution {
public int LengthOfLongestSubstring(string s) {
// 特殊情况处理
if (string.IsNullOrEmpty(s)) return 0; // 创建Dictionary存储字符最后出现的位置
Dictionary<char, int> lastPos = new Dictionary<char, int>();
int maxLength = 0;
int left = 0; // 右指针遍历字符串
for (int right = 0; right < s.Length; right++) {
char currentChar = s[right]; // 如果字符已存在,直接将左指针移动到重复字符的下一位
if (lastPos.ContainsKey(currentChar)) {
left = Math.Max(left, lastPos[currentChar] + 1);
} // 更新字符最后出现的位置
lastPos[currentChar] = right; // 更新最大长度
maxLength = Math.Max(maxLength, right - left + 1);
} return maxLength;
}
}

代码详解

基本滑动窗口版本:

  1. HashSet<char> window:用于存储当前窗口中的字符,保证无重复
  2. while (window.Contains(s[right])):当发现重复字符时,不断移动左指针并删除字符
  3. maxLength = Math.Max(maxLength, right - left + 1):更新最大长度

优化版本:

  1. Dictionary<char, int> lastPos:存储每个字符最后出现的位置
  2. left = Math.Max(left, lastPos[currentChar] + 1)
    • 当发现重复字符时,直接将左指针移动到重复字符上次出现位置的下一位
    • 使用Math.Max是为了防止左指针回退

执行结果

基本滑动窗口版本:

  • 执行用时:84 ms
  • 内存消耗:39.5 MB

优化版本:

  • 执行用时:72 ms
  • 内存消耗:39.8 MB

总结与反思

  1. 这是一道经典的滑动窗口题目,考察了:

    • 滑动窗口的基本应用
    • 字符串的处理
    • 哈希表的使用
  2. 两种实现方式各有优势:
    • 基本版本思路更直观,易于理解
    • 优化版本性能更好,避免了多次移动左指针
  3. 关键点:
    • 理解子串和子序列的区别
    • 正确维护滑动窗口
    • 高效处理重复字符

相关题目

  • LeetCode 第159题:至多包含两个不同字符的最长子串
  • LeetCode 第340题:至多包含 K 个不同字符的最长子串
  • LeetCode 第395题:至少有 K 个重复字符的最长子串
  • LeetCode 第424题:替换后的最长重复字符

LeetCode 第3题:无重复字符的最长子串的更多相关文章

  1. LeetCode 第三题--无重复字符的最长子串

    1. 题目 2.题目分析与思路 3.思路 1. 题目 输入: "abcabcbb" 输出: 3 解释: 因为无重复字符的最长子串是 "abc",所以其长度为 3 ...

  2. Leetcode(3)无重复字符的最长子串

    Leetcode(3)无重复字符的最长子串 [题目表述]: 给定一个字符串,请你找出其中不含有重复字符的 最长子串 的长度. 第一种方法:暴力 执行用时:996 ms: 内存消耗:12.9MB 效果: ...

  3. LeetCode刷题--无重复字符的最长子串(中等)

    题目描述: 给定一个字符串,请你找出其中不含有重复字符的 最长子串 的长度. 示例 1: 输入: "abcabcbb" 输出: 3 解释: 因为无重复字符的最长子串是 " ...

  4. 【LeetCode题解】3_无重复字符的最长子串(Longest-Substring-Without-Repeating-Characters)

    目录 描述 解法一:暴力枚举法(Time Limit Exceeded) 思路 Java 实现 Python 实现 复杂度分析 解法二:滑动窗口(双指针) 思路 Java 实现 Python 实现 复 ...

  5. leetcode题解#3:无重复字符的最长子串

    leetcode题解:无重复字符的最长子串 题目 给定一个字符串,请你找出其中不含有重复字符的 最长子串 的长度. 示例 1: 输入: s = "abcabcbb"输出: 3 解释 ...

  6. Leetcode(3)-无重复字符的最长子串

    给定一个字符串,找出不含有重复字符的最长子串的长度. 示例: 给定 "abcabcbb" ,没有重复字符的最长子串是 "abc" ,那么长度就是3. 给定 &q ...

  7. LeetCode 第 3 题:无重复字符的最长子串(滑动窗口)

    LeetCode 第 3 题:无重复字符的最长子串 (滑动窗口) 方法:滑动窗口 滑动窗口模板问题:右指针先走,满足了一定条件以后,左指针向前走,直到不满足条件. 特点:左右指针的方向是一致的,并且是 ...

  8. leetcode刷题第三天<无重复字符的最长子串>

    给定一个字符串,请你找出其中不含有重复字符的 最长子串 的长度. 示例 : 输入: "abcabcbb" 输出: 解释: 因为无重复字符的最长子串是 . 示例 : 输入: &quo ...

  9. leetcode刷题笔记-3. 无重复字符的最长子串(java实现)

    题目描述 给定一个字符串,请你找出其中不含有重复字符的 最长子串 的长度. 示例 1: 输入: "abcabcbb"输出: 3 解释: 因为无重复字符的最长子串是 "ab ...

  10. LeetCode随缘刷题之无重复字符的最长子串

    欢迎评论区交流. package leetcode.day_12_04; /** * 给定一个字符串 s ,请你找出其中不含有重复字符的最长子串的长度. * <p> * 示例1: * &l ...

随机推荐

  1. Codeforces Round 901 (Div

    C. Jellyfish and Green Apple 题解 显然\(n \% m =0\),答案一定为\(0\) 如果\(n > m\),我们显然可以将\(n / m\)的苹果分给每个人,然 ...

  2. 2024御网杯信息安全大赛个人赛wp_2024-11-27

    MISC题解 题目附件以及工具链接: 通过网盘分享的文件:御网杯附件 链接: https://pan.baidu.com/s/1LNA6Xz6eZodSV0Io9jGSZg 提取码: jay1 --来 ...

  3. C# 和 SQL Server中 PadLeft和PadRight 的用法

    C# 中 PadLeft和PadRight 的用法 需求:需要一个字符串实现自增.是根据数据库中一个自增的int类型的值,实现自增的.但是要加上前缀.比如,数据库中有一个自增的值,为,2.那么这个自增 ...

  4. MATLAB R2024a免费+破解版本(含密钥)

    强大功能 在使用2024a之前,我一直在使用2018b,最近数学建模+学年论文的摧残,让我看到了matlab新增的强大功能: Deep Learning Toolbox:新增支持 Transforme ...

  5. IO介绍-中

    系统接口 块设备接口 块设备:数据的存取和传输都是以数据块为单位的设备.典型的块设备是磁盘.该设备的基本特征是传输速率高,另一特征是可寻址,即能指定数据的输入源地址及输出的目标地址,可随机读写.磁盘设 ...

  6. SpringBoot重点详解--使用Actuator进行健康监控

    目录 添加依赖与配置 Actuator监控项 Actuator监控管理 打开或关闭 端口与地址 Actuator是Springboot提供的用来对应用系统进行自省和监控的功能模块,借助于Actuato ...

  7. 如何通过C#修改Windows操作系统时间

    C#的System.DateTime类提供了对日期时间的封装,用它进行时间的转换和处理很方便,但是我没有在其中找到任何可以用来修改系统时间的成员.用过VC.VB等的朋友可能知道,我们可以调用Win32 ...

  8. OpenGL ES 3.x游戏开发(上+下卷)书籍的电子版+源程序

    直接附上电子书以及源代码下载链接:https://pan.baidu.com/s/1G10hw5aIi-Bc2LyktwKrdg

  9. 即时通讯技术文集(第18期):IM架构设计基础知识合集 [共16篇]

    为了更好地分类阅读 52im.net 总计1000多篇精编文章,我将在每周三推送新的一期技术文集,本次是第18 期. [- 1 -] IM系统的MQ消息中间件选型:Kafka还是RabbitMQ? [ ...

  10. 飞书lark机器人 自动化发版

    飞书lark机器人 自动化发版 #1 介绍 开发飞书机器人接收消息并调用构建接口, 实现自动化发版 发送指令 -> 机器人接收指令 -> 调用jenkins-job远程构建与部署 jenk ...