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

  1. 无重复字符的最长子串

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

示例 1:

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

示例 2:

输入: s = "bbbbb"
输出: 1
解释: 因为无重复字符的最长子串是 "b",所以其长度为 1。

示例 3:

输入: s = "pwwkew"
输出: 3
解释: 因为无重复字符的最长子串是 "wke",所以其长度为 3。
     请注意,你的答案必须是 子串 的长度,"pwke" 是一个子序列,不是子串。

示例 4:

输入: s = ""
输出: 0

提示:

  • 0 <= s.length <= 5 * 10^4
  • s 由英文字母、数字、符号和空格组成

双指针算法

算法分析:

  • 定义一个HashMap<Character, Integer>,存放每个字母对应的个数,形成映射关系
  • 定义2个指针变量,i 和 j ,默认 i 在前,j 在后
  • 开始循环遍历,如果字母第一次出现,就把它对应的value值设置为1,这里采用了map.getOrDefault方法,它有2个参数,第一个是key,第二个是默认value值,如果map中key对应的没有value值,则设置为0,如代码所示,最后加1,表示当前增加一个
  • 如果map中key对应的value大于1,则表示出现了相同字母,这时候我们需要操控我们指针 j ,使当前key对应的value值变为1,则需要put(更新),使key对应的value值减1,一直到value的值等于1,即完成了对相同字母的剔除。此时会形成一个新的无循环字符串,长度为 i - j + 1
  • 最后进行取最大值(不懂的可以画图理解)

java代码

class Solution {
    public int lengthOfLongestSubstring(String s) {

       HashMap<Character,Integer> map = new HashMap<Character,Integer>();
       int res = 0;
       for(int i = 0,j = 0;i < s.length();i++){
           map.put(s.charAt(i),map.getOrDefault(s.charAt(i),0)+1);
           while(map.get(s.charAt(i))>1){
               map.put(s.charAt(j),map.get(s.charAt(j))-1);
               j++;
           }
           res = Math.max(res,i - j + 1);
       }
       return res;

    }
}

C++ 代码
算法
(双指针扫描) O(n)O(n)
定义两个指针 i,j(i<=j)i,j(i<=j),表示当前扫描到的子串是 [i,j][i,j] (闭区间)。扫描过程中维护一个哈希表unordered_map<char,int> hash,表示 [i,j][i,j]中每个字符出现的次数。
线性扫描时,每次循环的流程如下:

指针 jj 向后移一位, 同时将哈希表中 s[j]s[j] 的计数加一: hash[s[j]]++;
假设 jj 移动前的区间 [i,j][i,j] 中没有重复字符,则 jj 移动后,只有 s[j]s[j] 可能出现2次。因此我们不断向后移动 ii,直至区间 [i,j][i,j]中 s[j]s[j] 的个数等于1为止;
复杂度分析:由于 i,ji,j 均最多增加n次,且哈希表的插入和更新操作的复杂度都是 O(1)O(1),因此,总时间复杂度 O(n)

class Solution {
public:
    int lengthOfLongestSubstring(string s) {
        unordered_map<char, int> hash;
        int res = 0;
        for (int i = 0, j = 0; j < s.size(); j ++ )
        {
            hash[s[j]] ++ ;
            while (hash[s[j]] > 1) hash[s[i ++ ]] -- ;
            res = max(res, j - i + 1);
        }
        return res;
    }
};
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值