• 算法-3.无重复字符的最长子串


    题目

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

    示例

    示例1

    输入:s = "abcabcbb"
    输出: 3
    解释: 因为无重复字符的最长子串为"abc",所以其长度为 3
    
    • 1
    • 2
    • 3

    示例2

    输入: s = "bbbbbb"
    输出: 1
    解释:因为无重复字符的最长子串为 “b”,所以其长度为 1
    
    • 1
    • 2
    • 3

    示例3

    输入: s = "pwwkew"
    输出: 3
    解释:因为无重复字符的最长子串为"wke",所以其长度为 3.
    
    • 1
    • 2
    • 3

    思路

      以示例1的字符串abcabcbb为例,找出从每一个字符开始的,不包含重复字符的最长子串,那么其中最长的字符串即为答案。
      对于示例1中的字符串,列举以下结果,其中括号表示选中的字符以及最长的字符串:

    • (a)bcabcbb 开始的最长字符串为 (abc)abcbb;
    • a(b)cabcbb 开始的最长字符串为 a(bca)bcbb
    • ab©abcbb 开始的最长字符串为 ab(cab)cbb
    • abc(a)bcbb 开始的最长字符串为 abc(abc)bb
    • abca(b)cbb 开始的最长字符串为 abca(bc)bb
    • abcab©bb 开始的最长字符串为 abcab(cb)bb
    • abcabc(b)b 开始的最长字符串为 abcabc(b)b
    • abcabcb(b) 开始的最长字符串为 abcabcb(b)

      从上面数据中发现,如果依次递增地枚举子串的起始位置,那么子串的结束位置也是递增的!原因在于,假设选择字符串的第k个字符作为起始位置,并且得到了不包含重复字符的最长子串的结束位置为rk,那么当选择k+1到rk的字符显然是不重复的,并且由于少了原本的第k个字符,可以尝试继续增大rk,直到右侧出现了重复字符为止。
      这样一来,就可以使用「滑动窗口」来解决这个问题:

    • 使用两个指针表示字符串中的某个子串(或窗口)的左右边界,其中左指针代表着上文中枚举子串的起始位置,右指针即为上文中的rk;
    • 在每一步的操作中,会将左指针向右移动一格,表示 开始枚举下一个字符作为起始位置,然后可以不断的向右移动右指针,但需要保证这两个指针对应的子串中没有重复的字符。在移动结束后,这个子串就对应这以 左指针开始的,不包含重复字符的最长子串。记录下此子串的长度
    • 在枚举结束后,找到的最长的子串的长度即为答案。

    判断重复字符
      在上面流程中,还需要使用一种数据结构判断 是否有重复的字符,常用的数据结构为哈希集合(HashSet)。在左指针向右移动的时候,从哈希集合中移除一个字符,在右指针向右移动的时候,往哈希集合中添加一个字符。

    class Solution {
        public int lengthOfLongestSubstring(String s) {
            // 哈希集合,记录每个字符是否出现过
            Set<Character> occ = new HashSet<Character>();
            int n = s.length();
            // 右指针,初始值为 -1,相当于我们在字符串的左边界的左侧,还没有开始移动
            int rk = -1, ans = 0;
            for (int i = 0; i < n; ++i) {
                if (i != 0) {
                    // 左指针向右移动一格,移除一个字符
                    occ.remove(s.charAt(i - 1));
                }
                while (rk + 1 < n && !occ.contains(s.charAt(rk + 1))) {
                    // 不断地移动右指针
                    occ.add(s.charAt(rk + 1));
                    ++rk;
                }
                // 第 i 到 rk 个字符是一个极长的无重复字符子串
                ans = Math.max(ans, rk - i + 1);
            }
            return ans;
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23

    复杂度分析

    • 时间复杂度:O(N),其中N是字符串的长度。左指针和右指针分别会遍历整个字符串一次
    • 空间复杂度:O(∣Σ∣),其中Σ表示字符集(即字符串中可以出现的字符),∣Σ∣表示字符集的大小,在本题中没有明确说明字符集,因为默认为所有ASCII码在[0,128]内的字符,即∣Σ∣=128,需要用到哈希集合存储出现过的字符,而字符最多有∣Σ∣个,因此空间复杂度为O(∣Σ∣)
  • 相关阅读:
    计算机毕业设计Java化妆品销售网站(源码+系统+mysql数据库+lw文档)
    人工智能的未来发展前景:机遇与挑战
    共享wifi项目盘点:共享wifi加盟哪个品牌好?
    腾讯开源业界首个云原生标准的一站式微服务管理框架Femas
    【多式联运】基于帝国企鹅算法、遗传算法、粒子群算法求解多式联运路径优化问题附matlab代码
    更改搜索路径上的文件夹
    Django路由层解析
    记录--5个知识点,让 Vue3 开发更加丝滑
    【面试刷题】——C++的特点简单说明
    C进阶-数据的存储(下)
  • 原文地址:https://blog.csdn.net/qq_32530561/article/details/125554359