C.滑动窗口-求子数组个数-越短越合法——3134. 找出唯一性数组的中位数

题目链接:3134. 找出唯一性数组的中位数(困难)

算法原理:

解法:二分查找+滑动窗口

击败7.55%

时间复杂度O(Nlogn)

二分查找模板👇

优选算法-二分:18.在排序数组中查找元素的第一个和最后一个位置

①确定中位数位置:

nums 的唯一性数组的数量就是 nums 的非空连续子数组的个数

一共有m=1+2+⋯+n=2n(n+1)​ 个非空连续子数组

当 m=4 时,中位数是第 2 小元素,m=5 时,中位数是第 3 小元素

所以中位数是这 m 个数中的第k=(m+1)/2小

②二分查找调节mid找k(求最左端点模板)

以 "子数组的不同元素个数" 为二分对象(左边界 0、右边界 n),每次取中间值mid,判断 "不同元素数≤mid 的子数组数量" 是否≥k:

while(left<right)

1.分二段,发现左无= 右有 --->左无=就求左端点->check确定等号在右边,所以是求左端点

2.注意mid靠左:mid=left+(right-left)/2->确定mid咋写

3.if(mid在最左端点的左边)让左端点的左边往里靠,还要超过:left=mid+1->mid在最左端点的左边就是!check

4.else 另一边靠过来就行:right=mid

5.返回left或者right都行(因为最终会重合)

③用封装的check方法来判断是否覆盖第k小(ret>=k返回true)

维护窗口[left, right],保证窗口内不同元素数≤mid,累加合法子数组数(right-left+1),数量≥k 则提前返回 true

答疑

Q1:m既然是总的非空连续子数组的个数,那里面还有可能存在重复的元素呢

唯一性数组是:[1,1,1,1,2,2](长度 m=6,和子数组总数一致)------这里的元素可以重复,但数组长度固定为 m,中位数的位置只由长度 m 决定,和元素值是否重复无关

Java代码:

java 复制代码
class Solution {
    public int medianOfUniquenessArray(int[] nums) {
        int n=nums.length;
        long k=((long)n*(n+1)/2+1)/2;//中位数的位置,第k小的数
        //求左端点模板↓
        int left=0,right=n;
        while(left<right){
            int mid=left+(right-left)/2;//取靠左的位置
            //mid是不断调节的,目标位置在k
            //检查不同元素个数<=mid时的子数组数量是否>=k
            if(!check(nums,mid,k)) left=mid+1;//比k小,left往右调整
            else right=mid;//比k大,right往左调整
        }
        return right;
    }
    //检查是否覆盖第k小,ret>=k算覆盖到k
    private boolean check(int[] nums,int upper,long k){
        long ret=0;
        int n=nums.length,kinds=0;
        //<元素,个数>
        Map<Integer,Integer> hash=new HashMap<>();
        for(int left=0,right=0;right<n;right++){
            //进窗口
            int in=nums[right];
            if(hash.getOrDefault(in,0)==0) kinds++;
            hash.put(in,hash.getOrDefault(in,0)+1);
            //出窗口
            while(kinds>upper){
                int out=nums[left];
                hash.put(out,hash.get(out)-1);
                if(hash.get(out)==0){
                    hash.remove(out);
                    kinds--;
                }
                left++;
            }
            //更新
            ret+=right-left+1;
            //判断是否提前返回(无需遍历完所有子数组)
            if(ret>=k) return true;
        }
        return false;
    }
}
相关推荐
云烟成雨TD1 天前
Spring AI Alibaba 1.x 系列【6】ReactAgent 同步执行 & 流式执行
java·人工智能·spring
Wenweno0o1 天前
0基础Go语言Eino框架智能体实战-chatModel
开发语言·后端·golang
小O的算法实验室1 天前
2026年ASOC,基于深度强化学习的无人机三维复杂环境分层自适应导航规划方法,深度解析+性能实测
算法·无人机·论文复现·智能算法·智能算法改进
于慨1 天前
Lambda 表达式、方法引用(Method Reference)语法
java·前端·servlet
swg3213211 天前
Spring Boot 3.X Oauth2 认证服务与资源服务
java·spring boot·后端
gelald1 天前
SpringBoot - 自动配置原理
java·spring boot·后端
殷紫川1 天前
深入理解 AQS:从架构到实现,解锁 Java 并发编程的核心密钥
java
‎ദ്ദിᵔ.˛.ᵔ₎1 天前
LIST 的相关知识
数据结构·list
一轮弯弯的明月1 天前
贝尔数求集合划分方案总数
java·笔记·蓝桥杯·学习心得
chenjingming6661 天前
jmeter线程组设置以及串行和并行设置
java·开发语言·jmeter