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

点此看全部题解 LeetCode必刷100题:一份来自面试官的算法地图(题解持续更新中)

生活中的算法

你是否玩过"一指禅"游戏?就是沿着一串字母走,不能重复走过已经走过的字母。这个游戏的本质,其实就是在寻找无重复字符的最长子串。

在实际编程中,这个问题的应用非常广泛。比如在文本编辑器中查找不含重复字符的最长片段,或是在DNA序列分析中寻找无重复碱基对的最长序列。

问题描述

LeetCode第3题"无重复字符的最长子串"是这样描述的:给定一个字符串 s,请你找出其中不含有重复字符的最长子串的长度。

例如:

  • 输入: s = “abcabcbb”,输出: 3(最长子串是 “abc”)
  • 输入: s = “bbbbb”,输出: 1(最长子串是 “b”)
  • 输入: s = “pwwkew”,输出: 3(最长子串是 “wke”)

最直观的解法:暴力枚举法

最容易想到的方法是:枚举所有可能的子串,检查每个子串是否包含重复字符,然后找出最长的那个。

让我们用一个例子来模拟这个过程:

s = "pwwk"

检查所有子串:
"p" - 长度1,无重复
"pw" - 长度2,无重复
"pww" - 长度3,有重复
"pwwk" - 长度4,有重复
"w" - 长度1,无重复
"ww" - 长度2,有重复
"wwk" - 长度3,有重复
"w" - 长度1,无重复
"wk" - 长度2,无重复
"k" - 长度1,无重复 最长的无重复子串长度为2

这种思路可以用Java代码这样实现:

public int lengthOfLongestSubstring(String s) {
int maxLength = 0; // 枚举所有可能的起点
for (int i = 0; i < s.length(); i++) {
// 使用Set来检查是否有重复字符
Set<Character> charSet = new HashSet<>();
int currentLength = 0; // 从起点开始尝试延伸
for (int j = i; j < s.length(); j++) {
// 如果遇到重复字符,结束当前子串的检查
if (charSet.contains(s.charAt(j))) {
break;
}
charSet.add(s.charAt(j));
currentLength++;
} maxLength = Math.max(maxLength, currentLength);
} return maxLength;
}

优化解法:滑动窗口法

仔细观察会发现,当我们遇到重复字符时,不需要完全重新开始,而是可以从上一次该字符出现位置的下一个位置继续。这就是"滑动窗口"的思想。
举个例子,对字符串"abcdce",当我们查看到"abcd"这个子串时,子串内没有重复字符。
但是,当我们继续前进,子串变成"abcdc",现在c重复了!由于出现了第二个c,所以,第一个c之前的字符,都没有用了。
我们需要把第一个c,以及它前面的字符全部剔除出去,以保证c不再重复。

滑动窗口法的原理

  1. 使用两个指针(left和right)维护一个窗口
  2. 右指针不断向右移动,扩大窗口
  3. 当遇到重复字符时,左指针移动到上一次该字符出现位置的下一位
  4. 在这个过程中记录最大窗口大小

算法步骤(伪代码)

  1. 初始化left = 0,right = 0,maxLength = 0
  2. 使用Map记录每个字符最后出现的位置
  3. 移动右指针,对于每个字符:
    • 如果字符已在窗口中,更新left指针
    • 更新字符的位置
    • 更新最大长度

示例运行

让我们用s = "abba"模拟这个过程:

初始状态:left = 0, right = 0, maxLength = 0
Map = {} 1. 处理'a':
Map = {a:0}
窗口:[a]
maxLength = 1 2. 处理'b':
Map = {a:0, b:1}
窗口:[ab]
maxLength = 2 3. 处理'b':
发现重复的'b'
left移动到上一个'b'的下一位
Map = {a:0, b:2}
窗口:[b]
maxLength = 2 4. 处理'a':
发现重复的'a'
left移动到上一个'a'的下一位
Map = {a:3, b:2}
窗口:[ba]
maxLength = 2

Java代码实现

public int lengthOfLongestSubstring(String s) {
// 使用HashMap存储字符最后出现的位置
Map<Character, Integer> charMap = new HashMap<>();
int maxLength = 0; // left是窗口左边界,i是右边界
for (int left = 0, i = 0; i < s.length(); i++) {
char currentChar = s.charAt(i); // 如果字符已经在窗口中,更新左边界
if (charMap.containsKey(currentChar)) {
// 取max是为了防止left向左移动
left = Math.max(left, charMap.get(currentChar) + 1);
} // 更新字符位置和最大长度
charMap.put(currentChar, i);
maxLength = Math.max(maxLength, i - left + 1);
} return maxLength;
}

解法比较

让我们比较这两种解法:

暴力枚举法:

  • 时间复杂度:O(n²)
  • 空间复杂度:O(min(m,n)),其中m是字符集大小
  • 优点:直观易懂
  • 缺点:效率低,有重复计算

滑动窗口法:

  • 时间复杂度:O(n)
  • 空间复杂度:O(min(m,n))
  • 优点:一次遍历就能得到结果
  • 缺点:需要额外空间存储字符位置

题目模式总结

这道题体现了几个重要的算法思想:

  1. 滑动窗口:使用双指针维护一个符合条件的区间
  2. 空间换时间:使用哈希表记录信息来优化查找
  3. 重复利用信息:不重新开始,而是利用已知信息继续搜索

这种解题模式在很多问题中都有应用,比如:

  • 最小覆盖子串
  • 字符串的排列
  • 找到字符串中所有字母异位词

解决此类问题的通用思路是:

  1. 考虑是否可以通过维护一个窗口来解决
  2. 确定窗口的更新条件
  3. 想清楚如何移动左右指针
  4. 考虑是否需要额外的数据结构来优化

小结

通过这道题,我们不仅学会了如何找到最长无重复子串,更重要的是掌握了滑动窗口这个强大的算法技巧。从暴力解法到优化解法,我们看到了如何通过观察问题特点来优化算法。

记住,很多看似复杂的问题,都可以通过滑动窗口来优雅地解决。当你遇到类似的字符串处理问题时,不妨先想想是否可以用这个技巧!


作者:忍者算法
公众号:忍者算法

从一指禅到无重复字符:最长子串问题的优雅解法|LeetCode 3 无重复字符的最长子串的更多相关文章

  1. [LeetCode] Longest Substring Without Repeating Characters 最长无重复字符的子串

    Given a string, find the length of the longest substring without repeating characters. Example 1: In ...

  2. [LeetCode] Longest Substring Without Repeating Characters 最长无重复字符的子串 C++实现java实现

    最长无重复字符的子串 Given a string, find the length of the longest substring without repeating characters. Ex ...

  3. 【LeetCode】无重复字符的最长子串【滑动窗口法】

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

  4. LeetCode 3. 无重复字符的最长子串(Longest Substring Without Repeating Characters)

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

  5. 滑动窗口经典题 leetcode 3. 无重复字符的最长子串

    题目 解题思路 题目要求找出给定字符串中不含有重复字符的最长子串的长度.这是一个典型的滑动窗口的题目,可以通过滑动窗口去解答. 滑动窗口 具体操作如下图示:找到一个子串 s[left...right] ...

  6. 【Leetcode】无重复字符的最长子串

    暴力解法,枚举所有子字符串组合 输入:长度[0,n]的字符串 耗时过长--- class Solution { public: int lengthOfLongestSubstring(string ...

  7. LeetCode 3——无重复字符的最长子串

    1. 题目 2. 解答 2.1. 方法一 我们从前往后遍历字符串,start 代表最长子串的起始位置,一开始设置为零. 如果没有遇到重复字符,则更新子串的长度,向后遍历. 如果遇到重复字符时,则更新字 ...

  8. LeetCode OJ -- 无重复字符的最长子串

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

  9. [LeetCode] Longest Substring Without Repeating Characters 最长无重复子串

    Given a string, find the length of the longest substring without repeating characters. For example, ...

  10. 【LeetCode】无重复字符串最长子串

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

随机推荐

  1. 基于Ubuntu搭建Pwn调试环境

    Pwn环境配置 本文演示使用干净的Vmware下安装的的 Ubuntu 18.04 LTS镜像 配置以下Pwn环境: OS(系统)配置 VMware Tools net-tools open-vm-t ...

  2. npm : 无法加载文件 D:\Program Files\nodejs\npm.ps1,因为在此系统上禁止运行脚本

    升级node和npm之后,npm run dev 启动一个Vue项目,报错如下: npm : 无法加载文件 D:\Program Files\nodejs\npm.ps1,因为在此系统上禁止运行脚本. ...

  3. Golang之常用方法[总结]

    1. 有一堆数字,如果除了一个数字以外,其他数字都出现了两次,那么如何找到出现一次的数字? nums := []int{1, 5, 1, 6, 5, 3, 6} i := 0 for _, v := ...

  4. monitor磁盘空间不足警告

    虚拟机安装ceph时,执行ceph -s monitor主机遇到了 mon c101(monitor主机名) is low on available space 错误 这是我找到的解决办法 monit ...

  5. NET 6 中新增的LINQ 方法

    .NET 6 中添加了许多 LINQ 方法. 下表中列出的大多数新方法在 System.Linq.Queryable 类型中具有等效方法. 欢迎关注 如果你刻意练习某件事情请超过10000小时,那么你 ...

  6. 从Delphi到Lazarus——在Lazarus中使用ActiveX控件(导入类型库)

    0.前言 Lazarus的设计目标是应用Free Pascal,所以所有凡是Free Pascal能运行的平台,Lazarus都可以运行.通俗地说就是Lazarus追求的是跨平台编程,致力于" ...

  7. ng-alain: 配置开发环境

    配置 ng-alain 开发环境 安装 1. Yarn 官方文档实际上是基于 Yarn 1 的,请从 Yarn 1 开始.在创建项目之后,可以升级到 Yarn 3. 注意:直接通过 npm 安装 ya ...

  8. 2024年1月Java项目开发指南6:接口测试

    我们使用API Fox这款工具对接口进行测试. (你要是会其他的例如postman进行测试也行) https://apifox.com/ 新建一个项目,新增一个接口 因为这个接口没有参数,所以无需填写 ...

  9. 【Python】【爬虫】爬取小说5000章,遇到的爬虫问题与解决思路

    爬虫问题分析 回顾 之前写了一个爬取小说网站的多线程爬虫,操作流程如下: 先爬取小说介绍页,获取所有章节信息(章节名称,章节对应阅读链接),然后使用多线程的方式(pool = Pool(50)),通过 ...

  10. 【Linux】【虚拟机】 IP地址的动态与静态设置

    目录 配置文件的修改 配置文件的修改 vim /etc/sysconfig/network-scripts/ifcfg-ens33 IP配置方式(不指定:none,静态:static,动态:dhcp) ...