萌新记录~多多指教:)
来源:力扣(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]]<left 这个条件,如果成立则也可说明当前窗口中没有s[i]这个符号,应该添加进窗口中。
- 其实条件就两个:第一个是m[s[i]]==0说明没出现过,肯定可以添加进窗口
- 第二个是m[s[i]]!=0时,不能直接说不满足条件,因为前面left增加可能是因为它之后有了重复的元素。
参考代码
#include <iostream>
#include <string>
#include <cstring>
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;
}