至少有K个重复字符的最长子串

给你一个字符串 s 和一个整数 k ,请你找出 s 中的最长子串, 要求该子串中的每一字符出现次数都不少于 k 。返回这一子串的长度。

如果不存在这样的子字符串,则返回 0。

示例 1:

复制代码
输入:s = "aaabb", k = 3
输出:3
解释:最长子串为 "aaa" ,其中 'a' 重复了 3 次。

示例 2:

复制代码
输入:s = "ababbc", k = 2
输出:5
解释:最长子串为 "ababb" ,其中 'a' 重复了 2 次, 'b' 重复了 3 次。

思路:滑动窗口,没有条件滑就创造条件滑。规定 t 为滑动窗口内最大的字母种类(当取所有可能的 t 时,也就相当于全集了,等于没规定这个条件)。虽然最大种类数为 t 时,t-1可能也会被考虑一些次,但是一定会考虑到长度 仅为 t 的所有情况。而最后的res 一定是某个数目的 字母种类 ,所以一定可以考虑到。

设置 existNum、reachNum 两个变量,一个是存在字母种类数量,一个数到达k数目要求的字母种类数

python 复制代码
from collections import defaultdict


class Solution:
    def longestSubstring(self, s: str, k: int) -> int:
        charSet = set(s)
        res = 0
        # 滑动窗口内最大的字母种类
        for t in range(1,len(charSet)+1):
            exist=defaultdict(int)
            existNum=0
            reachNum=0
            l,r=0,0
            while r<len(s):
                exist[s[r]] += 1
                # 存在
                if exist[s[r]] == 1:
                    existNum += 1
                # 满足
                if exist[s[r]] == k:
                    reachNum += 1

                while l<=r and existNum>t:
                    exist[s[l]]-=1
                    if exist[s[l]]==0:
                        existNum-=1

                    if exist[s[l]] == k-1:
                        reachNum -= 1
                    l += 1

                if reachNum==existNum:
                    res=max(res,r-l+1)
                r += 1

自己写的一版错误的:

没有最外层的 t ,这样滑 处理不了 "baaabcb" 这种情况。

python 复制代码
from collections import defaultdict


class Solution:
    def longestSubstring(self, s: str, k: int) -> int:
        res = 0
        characterDict=defaultdict(int)
        for c in s:
            characterDict[c]+=1
        exist=defaultdict(int)
        existNum=0
        reachNum=0
        l,r=0,0
        while r<=len(s):
            if r==len(s) or exist[s[r]]+characterDict[s[r]]<k:
                while l<r and reachNum<existNum:
                    exist[s[l]]-=1
                    if exist[s[l]]==0:
                        existNum-=1

                    if exist[s[l]] == k-1:
                        reachNum -= 1
                    l += 1

                res=max(res,r-l)
                l=r=r+1

            else:
                exist[s[r]] += 1
                #存在
                if exist[s[r]]==1:
                    existNum+=1

                #满足
                if exist[s[r]]==k:
                    reachNum+=1
                r+=1
        return res
print(Solution().longestSubstring("baaabcb", 3))
相关推荐
To_OC7 小时前
LC 207 课程表:刚学图论那会儿,我连这是拓扑排序都没看出来
javascript·算法·leetcode
To_OC7 小时前
LC 208 实现 Trie 前缀树:曾被名字劝退,写完发现是送分题
javascript·算法·leetcode
BadBadBad__AK9 小时前
线段树维护区间 k 次方和
c++·数学·算法·stl
Warson_L14 小时前
Python `Annotated` 与 LangGraph Reducer 学习笔记
python
韩师傅14 小时前
海天线算法的前世今生
python·计算机视觉
韩师傅14 小时前
当你的甲方设备过烂,要如何快速出效果?
python·计算机视觉
Warson_L14 小时前
LangGraph的MessageState and HumanMessage
python
韩师傅15 小时前
当你的甲方吐槽天空不够蓝,你应该如何应对
python·计算机视觉
Warson_L15 小时前
python的类&继承
python
Warson_L16 小时前
类型标注/type annotation
python