10.滑动窗口解决:无重复字符的最长子串 | LeetCode 3 Java 题解

目录

一、题目解析

[示例 1:](#示例 1:)

[示例 2:](#示例 2:)

[示例 3:](#示例 3:)

二、算法原理

[解法一:暴力枚举 + 哈希表(O(n²))](#解法一:暴力枚举 + 哈希表(O(n²)))

解法二:滑动窗口(推荐!O(n))

滑动窗口的核心步骤:

三、编写代码(Java)

for循环写法(推荐!!!)

while循环写法

四、总结


OJ链接:无重复字符的最长子串

哈喽大家好!今天咱们来啃一道经典的算法题------无重复字符的最长子串。这道题在 LeetCode 上是第 3 题,难度中等,但绝对是必练的滑动窗口入门题!


一、题目解析

先看题目描述:

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

注意关键词:子串 (必须是连续的!)和无重复字符

示例 1:

  • 输入:s = "abcabcbb"

  • 输出:3

  • 解释:因为无重复字符的最长子串是 "abc",所以长度为 3。

示例 2:

  • 输入:s = "bbbbb"

  • 输出:1

  • 解释:因为无重复字符的最长子串是 "b",所以长度为 1。

示例 3:

  • 输入:s = "pwwkew"

  • 输出:3

  • 解释:因为无重复字符的最长子串是 "wke",所以长度为 3。(注意 "pwke"是子序列,不是子串!)

📌 小贴士:子串 vs 子数组 → 都是连续的一段!别被"子序列"带偏了!


二、算法原理

这道题有几种解法,我们重点讲两种:

解法一:暴力枚举 + 哈希表(O(n²))

  • 枚举所有子串,用哈希表判断是否有重复字符。

  • 时间复杂度高,不推荐,但可以作为理解题意的起点。

解法二:滑动窗口(推荐!O(n))

  • 利用"窗口"思想,维护一个++无重复字符的连续区间。++

  • 用两个指针 leftright表示窗口左右边界。

  • 用一个数组(或哈希表)记录每个字符出现的次数。

滑动窗口的核心步骤:

  1. 初始化left = 0, right = 0

  2. 进窗口 :让 s[right]进入窗口 → hash[s[right]]++

  3. 判断 :如果 hash[s[right]] > 1→ 说明有重复!

  4. 出窗口 :从左边开始删,直到重复字符被移除 → hash[s[left++]]--

  5. 更新结果ret = Math.max(ret, right - left + 1)

  6. 移动右指针right++

关键点:窗口内始终是无重复字符的!一旦重复,就收缩左边界,直到恢复无重复状态。


三、编写代码(Java)

我整理了两种写法,for循环写法更推荐(结构清晰,易读),也保留了while循环写法,方便你对比学习。

for循环写法(推荐!!!)

java 复制代码
class Solution {
    public int lengthOfLongestSubstring(String ss) {
        // 先转换为字符数组
        char[] s = ss.toCharArray();

        // 数组模拟哈希表存储每个字符出现的次数
        // 也是滑动窗口需要维护的数据
        int hash[] = new int[128];

        int n = ss.length();
        int len = 0;
        for(int left = 0, right = 0; right < n; right++) {
            // 入窗口
            hash[s[right]]++;

            // 判断
            while(hash[s[right]] > 1) {
                // 出窗口
                hash[s[left++]]--;
            }

            // 更新结果
            len = Math.max(len, right - left + 1);
        }
        return len;
    }
}

while循环写法

java 复制代码
class Solution {
    public int lengthOfLongestSubstring(String ss) {
        char[] s = ss.toCharArray();

        int[] hash = new int[128];

        int left = 0, right = 0;
        int ret = 0;
        int n = ss.length();

        while (right < n) {
            // 入窗口
            hash[s[right]]++;

            // 判断
            while (hash[s[right]] > 1) {
                // 出窗口
                hash[s[left++]]--;
            }

            // 更新结果
            ret = Math.max(ret, right - left + 1);
            // 让下一个字符进入窗口
            right++;
        }
        return ret;
    }
}

四、总结

这道题是滑动窗口的经典入门题,核心是:

  • 用双指针维护窗口

  • 用数组/哈希表记录字符频次

  • 遇到重复就收缩左边界

  • 每次更新最大长度

时间复杂度:O(n)

空间复杂度:O(1)(因为字符集固定128)

相关推荐
JieE21219 小时前
LeetCode 226. 翻转二叉树|JS 递归超详细拆解,二叉树入门经典题
javascript·算法
JieE21220 小时前
LeetCode 104. 二叉树的最大深度|递归思路超详细拆解
javascript·算法
用户3521802454751 天前
当 Prompt 学会"热更新":Spring Boot × Nacos3 AI 实战
java·spring boot·ai编程
vivo互联网技术1 天前
CVPR 2026 | 全新强化学习框架 BeautyGRPO:重塑真实人像
算法·大模型·cvpr·影像
Darling噜啦啦1 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
swipe1 天前
正则表达式入门到进阶:从表单校验到手写模板引擎
前端·javascript·面试
神奇小汤圆1 天前
RAG大厂面试题汇总:向量检索、混合检索、Rerank、幻觉处理高频问题
面试
东坡白菜1 天前
破局全栈:一个前端开发的Java入门实战记录(1)
java·全栈
唐青枫1 天前
Java Tomcat 实战指南:从 Servlet 容器到 Spring Boot 部署
java