395至少有K个重复字符的最长子串(分治法)

1、题目描述

找到给定字符串(由小写字符组成)中的最长子串 T , 要求 T 中的每一字符出现次数都不少于 k 。输出 T 的长度。

2、示例

输入:
s = "ababbc", k = 2

输出:
5

最长子串为 "ababb" ,其中 'a' 重复了 2 次, 'b' 重复了 3 次。

3、题解

基本思想:分治法,对于一个字符串来说,如果要求子串最少出现k次,那么如果某些字母出现的次数小于k,这些字母一定不会出现在最长的子串中,并且这些字母将整个字符子串分割成小段,这些小段有可能是最长的,但是由于被分割了,还是要检查这一小段,如果某些字母出现的次数小于k,会将小段继续分割下去,直到该小段所有字符出现次数都不小于k,即seg.empty(),就不可再分res=max(res,int(s.size()))

#include<iostream>
#include<vector>
#include<algorithm>
#include<unordered_set>
#include<unordered_map>
#include<set>
using namespace std;
class Solution {
public:
    int longestSubstring(string s, int k) {
        //基本思想:暴力超时
        if(s.size()<k)  return 0;
        int len=0;
        for(int i=0;i<=s.size()-k;i++)
        {
            for(int j=i+k-1;j<s.size()&&j-i+1>len;j++)
            {
                vector<int> cnt(26,0);
                for(int t=i;t<=j;t++)
                {
                    cnt[s[t]-'a']++;
                }
                bool flag=true;
                for(auto c:cnt)
                {
                    if(c>0&&c<k)
                    {
                        flag=false;
                        break;
                    }
                }
                if(flag)
                    len=max(len,j-i+1);
            }
        }
        return len;
    }
};
class Solution1 {
public:
    int res=0;
    int longestSubstring(string s, int k) {
        //基本思想:分治法,对于一个字符串来说,如果要求子串最少出现k次,那么如果某些字母出现的次数小于k,
        //这些字母一定不会出现在最长的子串中,并且这些字母将整个字符子串分割成小段,这些小段有可能是最长的
        //但是由于被分割了,还是要检查这一小段,如果某些字母出现的次数小于k,会将小段继续分割下去
        //直到该小段所有字符出现次数都不小于k,即seg.empty(),就不可再分res=max(res,int(s.size()))
        Recursion(s,k);
        return res;
    }
    void Recursion(string s,int k)
    {
        if(s.size()<k)  return;
        vector<int> cnt(26,0);
        set<char> seg;
        for(auto c:s)
            cnt[c-'a']++;
        for(int i=0;i<26;i++)
        {
            if(cnt[i]>0&&cnt[i]<k)
                seg.insert(i+'a');
        }
        if(seg.empty())
        {
            res=max(res,int(s.size()));
            return;
        }
        int start=0;
        for(int i=0;i<=s.size();i++)
        {
            if(i==s.size()||seg.find(s[i])!=seg.end())
            {
                Recursion(s.substr(start,i-start),k);
                start=i+1;
            }
        }
    }
};
int main()
{
    Solution1 solute;
    string s="ababbc";
    int k=2;
    cout<<solute.longestSubstring(s,k)<<endl;
    return 0;
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值