力扣-3-无重复字符的最长子串

一、题目理解

题目描述

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

示例
  • 示例 1:输入 s = "abcabcbb",输出 3(最长子串是 "abc",长度为 3);
  • 示例 2:输入 s = "bbbbb",输出 1(最长子串是 "b",长度为 1);
  • 示例 3:输入 s = "pwwkew",输出 3(最长子串是 "wke" 或 "kew",长度为 3)。
关键说明
  • 子串:是字符串中连续的字符序列(区别于子序列,子序列可以不连续);
  • 无重复:子串中每个字符都只能出现一次。

二、解题思路

这道题的最优解法是滑动窗口(双指针)+ 哈希表,核心逻辑是:

  1. 用两个指针(左指针 left、右指针 right)表示当前的"无重复子串窗口";
  2. 右指针向右移动,逐个加入字符,用哈希表(或数组)记录字符的最新位置;
  3. 若右指针遇到重复字符,且该字符在当前窗口内(位置 ≥ left),则将左指针移动到"重复字符的下一位",保证窗口内始终无重复;
  4. 每次移动右指针,计算当前窗口长度,更新最大长度。
思路对比(新手友好)
  • 暴力解法:枚举所有子串,判断是否有重复,时间复杂度 O(n2)O(n^2)O(n2),效率低;
  • 滑动窗口:仅遍历字符串一次,时间复杂度 O(n)O(n)O(n),是最优解。

三、Java代码实现(最优解)

java 复制代码
class Solution {
    public int lengthOfLongestSubstring(String s) {
        // 记录字符的最新索引(ASCII字符范围0-127,用数组比HashMap更高效)
        int[] charIndex = new int[128];
        // 初始化数组值为-1(表示字符未出现过)
        for (int i = 0; i < 128; i++) {
            charIndex[i] = -1;
        }
        
        int maxLen = 0; // 记录最长无重复子串长度
        int left = 0;   // 滑动窗口左指针(窗口起始位置)
        
        // 右指针right遍历字符串,逐个加入字符
        for (int right = 0; right < s.length(); right++) {
            char currChar = s.charAt(right);
            // 若当前字符已出现,且在当前窗口内(索引≥left),则移动左指针到重复字符的下一位
            if (charIndex[currChar] != -1 && charIndex[currChar] >= left) {
                left = charIndex[currChar] + 1;
            }
            // 更新当前字符的最新索引
            charIndex[currChar] = right;
            // 计算当前窗口长度,更新最大值
            maxLen = Math.max(maxLen, right - left + 1);
        }
        
        return maxLen;
    }
}

四、代码关键解析

  1. 数据结构选择(数组 vs HashMap)

    • 为什么用 int[128] 而不是 HashMap<Character, Integer>
      字符串的字符本质是ASCII码(范围0-127),数组访问速度比哈希表更快,且无需处理哈希冲突,空间复杂度仅 O(1)O(1)O(1)(数组大小固定为128)。
    • 数组值的含义:charIndex[c] 表示字符 c 最后一次出现的索引,初始值 -1 表示未出现。
  2. 滑动窗口核心逻辑

    s = "abcabcbb" 为例,分步拆解:

    • right=0(字符'a'):charIndex['a']=-1,窗口 [0,0],长度1,maxLen=1,更新 charIndex['a']=0
    • right=1(字符'b'):charIndex['b']=-1,窗口 [0,1],长度2,maxLen=2,更新 charIndex['b']=1
    • right=2(字符'c'):charIndex['c']=-1,窗口 [0,2],长度3,maxLen=3,更新 charIndex['c']=2
    • right=3(字符'a'):charIndex['a']=0 ≥ left=0,左指针移到 0+1=1,窗口 [1,3],长度3,maxLen仍为3,更新 charIndex['a']=3
    • 后续遍历:窗口会动态调整,但maxLen始终保持3,最终返回3。
  3. 关键判断条件
    charIndex[currChar] != -1 && charIndex[currChar] >= left

    • 第一个条件:字符是否出现过;
    • 第二个条件:字符的上一次出现位置是否在当前窗口内(比如字符'a'之前出现在索引0,但左指针已移到1,说明该重复字符不在窗口内,无需调整左指针)。
  4. 窗口长度计算
    right - left + 1:比如窗口是 [1,3],长度是 3-1+1=3(对应字符'b','c','a')。

五、边界情况测试

  • 空字符串:s="",返回0;
  • 单字符:s="a",返回1;
  • 全重复:s="bbbbb",返回1;
  • 无重复:s="abcd",返回4;
  • 重复字符在窗口外:s="abba",right=3(字符'a')时,charIndex['a']=0 < left=2,无需调整左指针,窗口 [2,3],长度2。

总结

  1. 核心解法:滑动窗口(双指针) 是解决"最长无重复子串"类问题的最优思路,时间复杂度 O(n)O(n)O(n)(仅遍历一次字符串);
  2. 优化技巧:用固定大小的数组 替代HashMap存储字符索引,兼顾时间和空间效率(空间复杂度 O(1)O(1)O(1));
  3. 关键细节:调整左指针时,必须保证左指针只向右移动(避免回退),且仅当重复字符在当前窗口内时才调整。

这道题是滑动窗口的入门经典题,掌握后可以举一反三解决类似的"子串/子数组"问题(比如"最小覆盖子串"),核心是理解"窗口的动态调整逻辑"。

相关推荐
寻寻觅觅☆9 小时前
东华OJ-基础题-106-大整数相加(C++)
开发语言·c++·算法
偷吃的耗子9 小时前
【CNN算法理解】:三、AlexNet 训练模块(附代码)
深度学习·算法·cnn
化学在逃硬闯CS10 小时前
Leetcode1382. 将二叉搜索树变平衡
数据结构·算法
ceclar12310 小时前
C++使用format
开发语言·c++·算法
Gofarlic_OMS11 小时前
科学计算领域MATLAB许可证管理工具对比推荐
运维·开发语言·算法·matlab·自动化
夏鹏今天学习了吗11 小时前
【LeetCode热题100(100/100)】数据流的中位数
算法·leetcode·职场和发展
忙什么果12 小时前
上位机、下位机、FPGA、算法放在哪层合适?
算法·fpga开发
董董灿是个攻城狮12 小时前
AI 视觉连载4:YUV 的图像表示
算法
ArturiaZ13 小时前
【day24】
c++·算法·图论
大江东去浪淘尽千古风流人物13 小时前
【SLAM】Hydra-Foundations 层次化空间感知:机器人如何像人类一样理解3D环境
深度学习·算法·3d·机器人·概率论·slam