力扣-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;
    }
}
相关推荐
万象.3 分钟前
redis通用命令与数据结构
数据结构·数据库·redis
凌乱风雨12117 分钟前
从源码角度解析C++20新特性如何简化线程超时取消
前端·算法·c++20
Jim-2ha015 分钟前
【平面几何】判断一个点是否在任意多边形的内部
算法
小宇的天下15 分钟前
Calibre : 一个简单的DRC rule file
数据结构
橘颂TA17 分钟前
【剑斩OFFER】算法的暴力美学——合并 k 个升序链表
算法·leetcode·牛客·结构与算法
前端小L17 分钟前
双指针专题(五):灵活的起跳——「无重复字符的最长子串」
javascript·算法·双指针与滑动窗口
C雨后彩虹23 分钟前
ReentrantLock入门:核心特性与基本使用
java·数据结构·reentrantlock·lock
爪哇部落算法小助手24 分钟前
每日两题day67
c++·算法
hk112424 分钟前
【BioTech/SystemArch】2026年度高可靠性医疗架构与生物遗传算法基准索引 (Benchmark Index)
算法·系统架构·数据集·生物信息学·垃圾回收
你撅嘴真丑27 分钟前
短信计费 和 甲流病人初筛
数据结构·c++·算法