面试高频算法题---无重复字符的最长子串

目录

🤿题目

🥅思路解析

🏒代码实现

🎿复杂度分析


🤿题目

题目链接:无重复字符的最长字串

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

🕐示例1:

输入:s = abcabcbb
输出:3
解释:abc,bca,bc,b,最长的无重复字符子串的长度为3

🕑示例2:

输入:s = bbbb
输出:1
解释:b为最长的无重复字串子串,长度为1

🕒示例3:

输入:s = pwwkew
输出:3
解释:pw,w,wke,最长无重复字符子串的长度为3

🥅思路解析

我们这里以abcabcbb为例

从上图中可以发现起始位置递增,结束位置也是递增,于是可以用滑动窗口的思路来解决这个问题,具体做法如下:

🍀1. 我们使用左右两个指针,左指针标记子串起始位置,右指针标记字串结束位置
🍂2. 左指针往右走的时候,我们发现子串会将前一个字符删除
🍀3. 右指针则继续从上一个字串结束的位置往右走
🍂4. 右指针往右遍历的规则是,后续字符在当前字串中不存在,并且右指针往右走的时候不能越界
🍀5. 当右指针满足条件时,将右指针每次遍历的字符都存在HashSet中,因为HashSet方便我们判断子串中是否有将要添加的字符
🍂6. 在左指针每次往右走之前将新的子串与上一个子串的长度进行比较,记录最大值 

🏒代码实现

class Solution {
    public int lengthOfLongestSubstring(String s) {
        Set<Character> m = new HashSet<>();
        //窗口问题
        //如果右边界开始给0,则[left,right),如果边界给-1,则[left,right]
        //这个会影响最后长度的比较  子串长度right-i         字串长度right-i+1
        int right = -1;   //字符串的右边界,刚开始最长字符串中没有字符,故边界给-1
        int size = s.length();   
        int count = 0;   //无重复字符串的长度字符串
        for(int i = 0;i < size;i++){  //i为左边界,依次朝右走
            if(i != 0){
                m.remove(s.charAt(i-1));  //每次左边界右移的时候,需要将前一个字符删除
            }
            //右边界不能越界,且s的下一个字符不在当前子串中
            while(right+1 < size && !m.contains(s.charAt(right+1))){ 
                m.add(s.charAt(right+1));  //向哈希set中添加
                right++;
            }
            count = Math.max(count,right-i+1); //更新长度
        }
        return count;
    }
}

🎿复杂度分析

🍻时间复杂度:左右指针分别遍历了一次字符串,故时间复杂度为O(2N),也就是O(N)

🥂空间复杂度:我们借助了HashSet这一数据结构,里面存储着无重复字符的字符,存储的长度最大为128,因为ASCLL码的范围为[0,128),故空间复杂度为O(M)M为字符的个数,M最大为128 

  • 4
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

X_H学Java

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值