最大中位数 3578

最大中位数
给定一个由 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×105,1≤k≤109,1≤ai≤109。

输入样例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

刚开始拿到这个题的时候,看了下数据有1e9,想到了二分,但是刚开始想到要使得中位数最大,应该得每一步都去考虑中位数左右和右边的情况(x)

正解:由于数字最大是1e9,极限情况有可能是1个数字本身是1e9,然后操作1e9次,这样得到的最大的书就是2e9了,所以二分左边界是0,右边界应该是2e9。每一次二分一个值,如果n/2以及其右边的数字有小于它 的,就把差值加起来,这样得到一个差值的和,和k进行比较,如果小于k,那么是一定可以得到这么大的中位数的,如果大于k的话,就得不到这个中位数。

然后每一次那个最右边的那个大于它的数可以使用upper_bound()这个函数,时间复杂度最大是O(nlogn)

#include
using namespace std;


int n,k;

const int N=2e5+10;

int a[N];
typedef long long LL;

bool check(int mid)
{
    LL ans=0;
    for(int i=n/2;i<(upper_bound(a,a+n,mid)-a);i++)
    {
        ans+=mid-a[i];
    }

    return ans<=k;
}

int main(){


    scanf("%d%d",&n,&k);

    for(int i=0;i<n;i++){
        scanf("%d",&a[i]);
    }
    
    sort(a,a+n);
    LL l=0,r=2e9+10;
    while(l<r)
    {
        int mid=l+r+1>>1;
        if(check(mid))l=mid;
        else r=mid-1;
    }

    printf("%lld",l);


    system("pause");


}

你可能感兴趣的:(二分,算法)