嘘~ 正在从服务器偷取页面 . . .

洛谷P10184 [YDOI R1] whk 题解


洛谷P10184 [YDOI R1] whk 题解

题目链接:P10184 [YDOI R1] whk

题意

小 Z 一共要卷 $n$ 门科目,第 $i$ 门科目他有且只有 $a_i$ 道题。有无数天时间,每天小 Z 可以做无数道题。

如果小 Z 认为一天是有趣的,仅当他在这一天至少做了 $t$ 门科目的题。

小 Z 想知道最多有多少天是有趣的。

输入格式

第一行,$2$ 个正整数 $n,t$。

接下来一行,有 $n$ 个整数,分别 $a_1,a_2,a_3,\dots,a_{n-1},a_n$。

输出格式

一个整数,输出小 Z 认为有趣的天数的最大值。

数据范围

$1\le t\le n\le5\times10^5$,$1\le a_i \le 10^6$。

一开始想到的是贪心,即用堆维护最多的 $t$ 种题,答案的增量为这 $t$ 个中最小的那个。

不过这样的复杂度是 $\mathcal{O}(nt\log n)$ 的,因为每次减去最小值最少只会去掉一个种类

实际上,我们只需要二分一下这个天数 $x$ 就可以判断了,对于每个 $a_i$ 它至多用 $\max(a_i,x)$ 次

时间复杂度 $\mathcal{O}(n \log nV)$

代码:

#include <bits/stdc++.h>
using namespace std;
#define int long long
#define INF 0x3f3f3f3f3f3f3f3f
void up(int &x,int y) { x < y ? x = y : 0; }
void down(int &x,int y) { x > y ? x = y : 0; }
#define N ((int)(5e5 + 15))

int n, t, l, r, a[N];
bool check(int x)
{
    int sum = 0;
    for(int i = 1; i <= n; i++) sum += min(a[i], x);
    return sum >= x * t;
}
signed main()
{
    ios::sync_with_stdio(0);
    cin.tie(0);cout.tie(0);
    // freopen("check.in","r",stdin);
    // freopen("check.out","w",stdout);
    cin >> n >> t;
    for(int i = 1; i <= n; i++) cin >> a[i], r += a[i];
    while(l < r)
    {
        int mid = (l + r + 1) >> 1;
        check(mid) ? l = mid : r = mid - 1;
    }
    cout << l << '\n';
    return 0;
}

题外话

双倍经验:[ABC227D] Project Planning ,不过需要加一个 __int128_t


文章作者: q779
版权声明: 本博客所有文章除特别声明外,均采用 CC BY-NC-ND 4.0 许可协议。转载请注明来源 q779 !
评论
  目录