Leetcode--003--无重复字符的最长字串【C++、滑动窗口】

萌新记录~多多指教:)
来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/longest-substring-without-repeating-characters
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

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

示例 1:
输入: “abcabcbb”
输出: 3
解释: 因为无重复字符的最长子串是 “abc”,所以其长度为 3。

示例 2:
输入: “bbbbb”
输出: 1
解释: 因为无重复字符的最长子串是 “b”,所以其长度为 1。

示例 3:
输入: “pwwkew”
输出: 3
解释: 因为无重复字符的最长子串是 “wke”,所以其长度为 3。

请注意,你的答案必须是 子串 的长度,“pwke” 是一个子序列,不是子串。

思路:这是尝试滑动窗口的第一题
对于字符串 s

  • STEP1 定义一个空数组m[256],左界left=0,右界通过i 循环s。这里是因为ASCII总共能表示256个字符,所以可以记录所有出现过的字符。
  • STEP2 通过循环遍历字符串s,判断m[s[i]]是否为0,如果为0说明以前没有出现过,left不变,更新res值,更新m[s[i]]值。这里因为s下标从0开始,因此计算长度res时,用i-left后还应该+1;例如abcd left=0,i=3时都没有重复字符,此时res=i-left+1=3-0+1=4.
  • STEP3 如果m[s[i]]!=0 说明当前滑动窗口中含有相同元素,则缩小左界 left=m[s[i]];更新m[s[i]];这个m[s[i]]数组存储的是最近一次s[i]出现的位置的下一个下标。(因为我们本质是当出现重复元素时,就在滑动窗口中删除上一次出现的元素,并把left置为下一个位置)
  • STEP4 还应该注意另一种情况 例如abbca 当i=4时 此时数组m[a]=1 m[b]=3 m[c]=4 此时的left=2 m[s[4]]=m[a]=1 此时是不为0 但是也应该添加进滑动窗口中,因此在判断时还应该判断m[s[i]]
  • 其实条件就两个:第一个是m[s[i]]==0说明没出现过,肯定可以添加进窗口
  • 第二个是m[s[i]]!=0时,不能直接说不满足条件,因为前面left增加可能是因为它之后有了重复的元素。

参考代码

#include 
#include 
#include 

using namespace std;

int LengthofLongeststring(string s)
{
    int m[256];
    memset(m,0,sizeof(m));

    int res=0,left=0;
    for(int i=0;i<s.size();i++)
    {
        if(m[s[i]]==0 || m[s[i]]<left)
        {
            res=max(res,i-left+1);
        }
        else
            left=m[s[i]];
        m[s[i]]=i+1;
    }
    return res;
}
int main()
{
    string s;
    getline(cin,s);

    int ans;
    ans=LengthofLongeststring(s);
    cout << ans <<endl;
    return 0;
}

你可能感兴趣的:(编程题练习,leetcode)