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

目录

题目

[解题思路:滑动窗口 + 哈希表](#解题思路:滑动窗口 + 哈希表)

核心思想

详细实现步骤

图解示例

时间和空间复杂度

正确的写法


题目

3. 无重复字符的最长子串 - 力扣(LeetCode)

解题思路:滑动窗口 + 哈希表

滑动窗口是处理子串/子数组问题的常用技巧,结合哈希表可以高效解决此问题。

核心思想

维护一个"窗口",这个窗口内的所有字符都是不重复的。当遇到重复字符时,调整窗口左边界,确保窗口内无重复字符。

详细实现步骤

  1. 初始化:
  • 创建哈希表记录字符出现次数
  • 左指针left和右指针right初始化为0
  • 最大长度ret初始化为0
  1. 扩展窗口:
  • 右指针right向右移动,将当前字符加入窗口
  • 更新哈希表中该字符的计数
  1. 处理重复:
  • 如果当前字符在窗口中已存在(计数>1)
  • 不断移动左指针left,同时减少对应字符的计数
  • 直到窗口中不再有重复字符
  1. 更新结果:
  • 每次调整窗口后,计算当前无重复子串的长度
  • 更新最大长度ret
  1. 继续扩展:
  • 右指针继续向右移动,重复步骤2-4

图解示例

以字符串 "abcabcbb" 为例:

初始状态:

XML 复制代码
字符串: a b c a b c b b
        ↑
       l,r
哈希表: {}
最大长度: 0

步骤1:右指针移动到'a'

XML 复制代码
字符串: a b c a b c b b
        ↑
       l r
哈希表: {a:1}
最大长度: 1

步骤2:右指针移动到'b'

XML 复制代码
字符串: a b c a b c b b
        ↑ ↑
        l r
哈希表: {a:1, b:1}
最大长度: 2

步骤3:右指针移动到'c'

XML 复制代码
字符串: a b c a b c b b
        ↑   ↑
        l   r
哈希表: {a:1, b:1, c:1}
最大长度: 3

步骤4:右指针移动到第二个'a'

XML 复制代码
字符串: a b c a b c b b
        ↑     ↑
        l     r
哈希表: {a:2, b:1, c:1}

发现'a'重复,移动左指针直到窗口中'a'不重复:

XML 复制代码
字符串: a b c a b c b b
          ↑   ↑
          l   r
哈希表: {a:1, b:0, c:0}
最大长度: 3

步骤5:右指针移动到第二个'b'

XML 复制代码
字符串: a b c a b c b b
          ↑     ↑
          l     r
哈希表: {a:1, b:1, c:0}
最大长度: 3

步骤6:右指针移动到第二个'c'

XML 复制代码
字符串: a b c a b c b b
          ↑       ↑
          l       r
哈希表: {a:1, b:1, c:1}
最大长度: 3

步骤7:右指针移动到第三个'b'

XML 复制代码
字符串: a b c a b c b b
          ↑         ↑
          l         r
哈希表: {a:1, b:2, c:1}

'b'重复,移动左指针:

XML 复制代码
字符串: a b c a b c b b
                ↑   ↑
                l   r
哈希表: {a:0, b:1, c:0}
最大长度: 3

步骤8:右指针移动到第四个'b'

XML 复制代码
字符串: a b c a b c b b
                ↑     ↑
                l     r
哈希表: {a:0, b:2, c:0}

'b'重复,移动左指针:

XML 复制代码
字符串: a b c a b c b b
                    ↑ ↑
                    l r
哈希表: {a:0, b:1, c:0}
最大长度: 3

最终结果:最大无重复子串长度为3

时间和空间复杂度

  • 时间复杂度:O(n),其中n是字符串长度。每个字符最多被访问两次(右指针遍历和左指针调整)。
  • 空间复杂度:O(min(m,n)),其中m是字符集大小,n是字符串长度。哈希表最多存储min(m,n)个字符。

正确的写法

cpp 复制代码
class Solution {
public:
    int lengthOfLongestSubstring(string s) {
        unordered_map<char,int> hash;  //这里可以使用数组代替哈希表可以提高效率
        int left=0;
        int right=0;
        int ret=0;

        while(right < s.size())
        {
            hash[s[right]]++;
            while(hash[s[right]]>1)
            {
                hash[s[left++]]--;
            }

            right++;
            ret = max(ret,right-left);

            
        }
        return ret;
        
    }
};
相关推荐
We་ct2 小时前
LeetCode 5. 最长回文子串:DP + 中心扩展
前端·javascript·算法·leetcode·typescript
王老师青少年编程6 小时前
csp信奥赛C++高频考点专项训练之贪心算法 --【哈夫曼贪心】:合并果子
c++·算法·贪心·csp·信奥赛·哈夫曼贪心·合并果子
叼烟扛炮7 小时前
C++第二讲:类和对象(上)
数据结构·c++·算法·类和对象·struct·实例化
天疆说7 小时前
【哈密顿力学】深入解读航天器交会最优控制中的Hamilton函数
人工智能·算法·机器学习
wuweijianlove8 小时前
关于算法设计中的代价函数优化与约束求解的技术7
算法
leoufung8 小时前
LeetCode 149: Max Points on a Line - 解题思路详解
算法·leetcode·职场和发展
样例过了就是过了8 小时前
LeetCode热题100 最长公共子序列
c++·算法·leetcode·动态规划
HXDGCL9 小时前
矩形环形导轨:自动化循环线的核心运动单元解析
运维·算法·自动化
谭欣辰9 小时前
C++ 排列组合完整指南
开发语言·c++·算法
代码中介商9 小时前
银行管理系统的业务血肉 —— 流程、状态机、输入校验与持久化(下篇)
c语言·算法