力扣-2831

题目

给你一个下标从 0 开始的整数数组 nums 和一个整数 k

如果子数组中所有元素都相等,则认为子数组是一个 等值子数组 。注意,空数组是 等值子数组

nums 中删除最多 k 个元素后,返回可能的最长等值子数组的长度。

子数组 是数组中一个连续且可能为空的元素序列。

示例 1:

复制代码
输入:nums = [1,3,2,3,1,3], k = 3
输出:3
解释:最优的方案是删除下标 2 和下标 4 的元素。
删除后,nums 等于 [1, 3, 3, 3] 。
最长等值子数组从 i = 1 开始到 j = 3 结束,长度等于 3 。
可以证明无法创建更长的等值子数组。

示例 2:

复制代码
输入:nums = [1,1,2,2,1,1], k = 2
输出:4
解释:最优的方案是删除下标 2 和下标 3 的元素。 
删除后,nums 等于 [1, 1, 1, 1] 。 
数组自身就是等值子数组,长度等于 4 。 
可以证明无法创建更长的等值子数组。

思路

用一个滑动窗口记录最长的子序列,包括要删除的元素,在窗口内统计每个数的个数,使得窗口内的元素个数减去等值元素的个数,结果不超过 k。那么我们只需要求出最大的满足条件的窗口即可

解题方法

用下标i,j表示左右端点,用j<n作为边界值,保证每一个元素都会遍历到,统计每个元素的个数,当数组要删除的元素大于k。i++并更新元素个数,用ans保存最大的元素个数(此时元素一定是符合要求的,求最大就行)

代码

java 复制代码
class Solution {
    public int longestEqualSubarray(List<Integer> nums, int k) {
        int n=nums.size();
        int ans=0;
        Map<Integer,Integer> map=new HashMap<>();
        for(int i=0,j=0;j<n;j++){
            map.put(nums.get(j), map.getOrDefault(nums.get(j), 0) + 1);
            while(j-i+1-map.get(nums.get(i))>k){
                map.put(nums.get(i),map.get(nums.get(i))-1);
                i++;
            }
            ans=Math.max(ans,map.get(nums.get(j)));
        }
        return ans;
    }
}
相关推荐
Fanxt_Ja3 小时前
【LeetCode】算法详解#15 ---环形链表II
数据结构·算法·leetcode·链表
侃侃_天下3 小时前
最终的信号类
开发语言·c++·算法
茉莉玫瑰花茶3 小时前
算法 --- 字符串
算法
博笙困了4 小时前
AcWing学习——差分
c++·算法
NAGNIP4 小时前
认识 Unsloth 框架:大模型高效微调的利器
算法
NAGNIP4 小时前
大模型微调框架之LLaMA Factory
算法
echoarts4 小时前
Rayon Rust中的数据并行库入门教程
开发语言·其他·算法·rust
Python技术极客4 小时前
一款超好用的 Python 交互式可视化工具,强烈推荐~
算法
徐小夕4 小时前
花了一天时间,开源了一套精美且支持复杂操作的表格编辑器tablejs
前端·算法·github
小刘鸭地下城4 小时前
深入浅出链表:从基础概念到核心操作全面解析
算法