2021-02-02:力扣424. 替换后的最长重复字符。如何用代码实现?
福哥答案2021-02-02:
双指针
我们可以枚举字符串中的每一个位置作为右端点,然后找到其最远的左端点的位置,满足该区间内除了出现次数最多的那一类字符之外,剩余的字符(即非最长重复字符)数量不超过 kk 个。
这样我们可以想到使用双指针维护这些区间,每次右指针右移,如果区间仍然满足条件,那么左指针不移动,否则左指针至多右移一格,保证区间长度不减小。
虽然这样的操作会导致部分区间不符合条件,即该区间内非最长重复字符超过了 kk 个。但是这样的区间也同样不可能对答案产生贡献。当我们右指针移动到尽头,左右指针对应的区间的长度必然对应一个长度最大的符合条件的区间。
实际代码中,由于字符串中仅包含大写字母,我们可以使用一个长度为 2626 的数组维护每一个字符的出现次数。每次区间右移,我们更新右移位置的字符出现的次数,然后尝试用它更新重复字符出现次数的历史最大值,最后我们使用该最大值计算出区间内非最长重复字符的数量,以此判断左指针是否需要右移即可。
注意点:
1.滑动窗口只会变大。不会变小。
2.每循环一次,右指针一定右滑一次。左指针可能右滑一次,可能不滑动。
3.最大字符数,是各个历史窗口的最大字符数。
代码用golang编写,代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | func characterReplacement(s string, k int) int { sLen := len(s) //记录次数的字典表 dict := make([]int, 256) //记录窗口的最大字符数,可能是历史窗口最大字符数 maxCount := 0 L := 0 for R := 0; R < sLen; R++ { dict[s[R]]++ if maxCount < dict[s[R]] { maxCount = dict[s[R]] } //左指针要么右滑一次,要么不滑。也就是说,滑动窗口只会变大,不会变小 if R-L+1-maxCount > k { dict[s[L]]-- L++ } } return sLen - L } |
执行结果如下:
***
[力扣:424. 替换后的最长重复字符](https://leetcode-cn.com/problems/longest-repeating-character-replacement/)
[评论](https://user.qzone.qq.com/3182319461/blog/1612220668)
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· winform 绘制太阳,地球,月球 运作规律
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· AI 智能体引爆开源社区「GitHub 热点速览」
· Manus的开源复刻OpenManus初探
· 写一个简单的SQL生成工具