3578 最大中位数(二分查找)

1. 问题描述:

给定一个由 n 个整数组成的数组 a,其中 n 为奇数。你可以对其进行以下操作:
选择数组中的一个元素(例如 ai),将其增加 1(即,将其替换为 ai+1)。
你最多可以进行 k 次操作,并希望该数组的中位数能够尽可能大。奇数长度的数组的中位数是数组以非降序排序后的中间元素。例如,数组 [1,5,2,3,5] 的中位数为 3。

输入格式

第一行包含两个整数 n 和 k。第二行包含 n 个整数 a1,a2,…,an。

输出格式

输出一个整数,表示通过操作可能得到的最大中位数。

数据范围

对于 30% 的数据,1 ≤ n ≤ 5。
对于 100% 的数据,1 ≤ n ≤ 2 × 10 ^ 5,1 ≤ k ≤ 10 ^ 9,1 ≤ ai ≤ 10 ^ 9。

输入样例1:

3 2
1 3 5

输出样例1:

5

输入样例2:

5 5
1 2 1 1 1

输出样例2:

3

输入样例3:

7 7
4 1 2 4 3 4 4

输出样例3:

5
来源:https://www.acwing.com/problem/content/3581/

2. 思路分析:

分析题目可以知道题目是具有单调性的(看到题目数据范围为10 ^ 5而且在某个限制条件下的最值问题需要考虑能否使用二分来解决),当中位数越大的时候那么需要的操作次数是越多的,如果当前枚举的答案为ans,ans左边的数字都是满足题目要求的,ans右边的数字是不满足要求的,所以我们可以使用二分来解决;我们可以枚举答案,也即枚举中位数,枚举的区间范围为[1,2e9],判断当前数组a的中位数变为当前枚举的mid是否可以在小于等于k次操作内完成,如果可以说明中位数大于等于当前枚举的mid,否则中位数一定小于mid,根据check函数的结果来更新下一次二分的区间范围。

3. 代码如下:

from typing import List


class Solution:
    # 判断将a的中位数变为x的操作次数是否小于等于k, 计算中间位置到a的末尾的位置变为x需要的操作次数count
    def check(self, x: int, k: int, a: List[int]):
        count = 0
        for i in range(len(a) // 2, len(a)):
            if a[i] < x:
                count += x - a[i]
        return count <= k

    def process(self):
        n, k = map(int, input().split())
        a = list(map(int, input().split()))
        # 从小到大排序
        a.sort()
        # 注意中位数可能是1e9, 所以r要设置为2e9
        res = 0
        l, r = 1, 2 * 10 ** 9
        while l <= r:
            mid = l + r >> 1
            if self.check(mid, k, a):
                res = mid
                l = mid + 1
            else:
                r = mid - 1
        return res


if __name__ == '__main__':
    print(Solution().process())
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值