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

1. 题目

2. 解答
2.1. 方法一

我们从前往后遍历字符串,start 代表最长子串的起始位置,一开始设置为零。

如果没有遇到重复字符,则更新子串的长度,向后遍历。

如果遇到重复字符时,则更新字符串起始位置为上一个相同字符的后面一个位置,同时更新子串长度。

重复上面这个过程,直到遍历完毕。

'abcabc',start = 0,str_len = 1, 2, 3 此时第二次遇到 'a',start = 1,str_len = 3 此时第二次遇到 'b',start = 2,str_len = 3 此时第二次遇到 'c',start = 3,str_len = 3

class Solution:
    def lengthOfLongestSubstring(self, s):
        """
        :type s: str
        :rtype: int
        """

        max_len = 0
        str_len = 0
        start = 0      # 最长子串的起始位置
        index = 0     # 上一个相同字符在子串中的位置,是一个相对位置,不是在原字符串中的位置
        
        for i in range(len(s)):
            
            if (s[i] not in s[start:i]):
                str_len += 1
               
            # 如果遇到重复字符,更新子串的起始位置为上一个相同字符的后面一个位置
            # 同时我们需要更新子串长度
            else:
                max_len = max(max_len, str_len)
                index = s[start:i].find(s[i])
                str_len = str_len - index
                start = start + index + 1 
                
        max_len = max(max_len, str_len) # 一直没有遇到重复字符
        return max_len
复制代码
2.2. 方法二

方法一中,我们每次判断当前字符是否为重复字符时,都需要在子串中进行搜索,更新子串起始位置时,也要在子串中搜索上一个相同字符的位置,效率很低。

其实我们需要知道的就是一个子串的起始位置,然后往后遍历的时候只需要在适当的时候更新这个起始位置重新计算子串长度即可。

因此,我们可以建立一个当前字符和当前字符下一个位置的映射。

所有映射全部初始化为零,start = 0。从前往后开始遍历字符串,同时更新映射,计算子串长度。

如果当前字符的映射大于 start,说明在 satrt 后面出现过当前字符,就更新 start

class Solution:
    def lengthOfLongestSubstring(self, s):
        """
        :type s: str
        :rtype: int
        """
        max_len = 0
        str_len = 0
        start = 0    # 最长子串的起始位置
        index = 0   # 重复的字符在子串中的位置
        
        # 初始化映射
        table = []
        for i in range(128):
            table.append(0)
                
        for i in range(len(s)):
            
            start =  max(start, table[ord(s[i])])

            str_len = i - start + 1    
            max_len = max(max_len, str_len)
            
            table[ord(s[i])] = i + 1
            
        return max_len
复制代码
class Solution {
public:
    int lengthOfLongestSubstring(string s) {
        
        int table[128] = {0}; // 自动初始化为零
        int max_len = 0;
        int str_len = 0;
        int start = 0;
        
        string::iterator it = s.begin();
            
        for (int j = 0; it != s.end(); it++, j++)
        {
            start = start > table[*it] ? start : table[*it];
            table[*it] = j + 1;
            str_len = j - start + 1;
            max_len = max_len < str_len ? str_len : max_len;
        }
        
        return max_len;
    }
};
复制代码

获取更多精彩,请关注「seniusen」!

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值