无重复字符的最长子串
给定一个字符串,请你找出其中不含有重复字符的 最长子串 的长度。
示例 1:
输入: “abcabcbb”
输出: 3
解释: 因为无重复字符的最长子串是 “abc”,所以其长度为 3。
解法
首先是子串,所以连续的。一个滑动窗口,窗口内的都是没有重复的字符,我们需要尽可能的扩大窗口的大小。由于窗口在不停向右滑动,窗口的右边界就相当于遍历到的字符的位置。 为了求出窗口的大小,我们需要一个变量 left 来指向滑动窗口左边界,一个变量记录窗口的右边界如果该字符没有出现过就右边界增加,否则就移动左边界直到重复的字符去掉。 为了判断字符是否出现过把出现过的字符都放入 set 中,遇到 set 中没有的字符就加入 set 中并更新结果 res,如果遇到重复的,则从左边开始删字符,直到删到重复的字符停止。
AC代码
class Solution {
public:
int lengthOfLongestSubstring(string s) {
set<char>ss;
int ans=0;
int l=0,r=0;
while(r<s.size())
{
if(!ss.count(s[r]))
{
ss.insert(s[r++]);
ans=max(ans,(int)ss.size());
}
else
ss.erase(s[l++]);
}
return ans;
}
};
至多k个无重复字符的最长子串
#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <vector>
#include <queue>
#include <stack>
#include <set>
#include <map>
#include <sstream>
using namespace std;
typedef long long LL;
const int maxn=110;
const int MOD=1e9+7;
int k;
string s;
int main(){
cin>>k;
cin>>s;
int ans=0;
int l=0,r=0;
map<char,int>m;
for(int i=0;i<s.length();i++){
m[s[i]]++;
r=i;
while(m.size()>k){
m[s[l]]--;
if(m[s[l]]==0)
m.erase(m.find(s[l]));
l++;
}
ans=max(ans,r-l+1);
}
cout<<ans<<endl;
return 0;
}
至少有K个重复字符的最长子串
找到给定字符串(由小写字符组成)中的最长子串 T , 要求 T 中的每一字符出现次数都不少于 k 。输出 T 的长度。
输入:
s = “ababbc”, k = 2
输出:
5
最长子串为 “ababb” ,其中 ‘a’ 重复了 2 次, ‘b’ 重复了 3 次。
解法
使用两个指针 left、right 和小写字母计数器统计字符段中各个字符出现的次数,然后调整当前的 left、right 指针。再遍历一遍修正后的字符段,如果中间有出现次数小于 k 的字符,那么将修正后的字符段拆开为 [left, index - 1] 和 [index + 1, right] 两个字符段重新寻找。
AC代码
class Solution {
public:
int longestSubstring(string s, int k) {
return myHelper(s, k, 0, s.size() - 1);//left == 0, right = s.size() - 1
}
//寻找s串中[left, right]字符段中至少有K个重复数字的最长子串
int myHelper(string