【每日一题】补档 CF1065 C. Make It Equal | 思维 | 中等

题目内容

原题链接

给定一个长度为 n n n 的数组 a a a ,每次操作可以选择一个数 x x x ,将所有大于 x x x 的数都下降为 x x x ,一次操作的下降总代价为 s s s ,要求 s ≤ k s\leq k s≤k ,问需要多少次操作使得数组 a a a 的所有数都相同。

数据范围

  • 1 ≤ n ≤ 2 ⋅ 1 0 5 1\leq n\leq 2\cdot 10^5 1≤n≤2⋅105
  • n ≤ k ≤ 1 0 9 n\leq k\leq 10^9 n≤k≤109
  • 1 ≤ a i ≤ 2 ⋅ 1 0 5 1\leq a_i\leq 2\cdot 10^5 1≤ai≤2⋅105

题解

朴素做法

首先使得所有数都相同,则使得所有值都成为 m i n ( a ) min(a) min(a) 。

所以考虑怎么将所有数都变成 m i n ( a ) min(a) min(a) 。

先对 a a a 排序,然后从最大的数开始依次将前面大的数往小了降。

从大的值开始往下依次枚举,看每次的 k k k 可以将前面多少数统一下降为一个数。

因为值域很小,所以可以直接枚举值域,这样的时间复杂度是 O ( m a x ( a ) + n ) O(max(a)+n) O(max(a)+n)

优化做法

如果值域很大,是否有与值域无关的做法。

其实这种题本身就应该值域无关,考虑当前最大的数为 c u r cur cur ,有 c n t cnt cnt 个值为 c u r cur cur 的数,下一步变成一个大于 a [ j ] a[j] a[j] 的最小的数 n x t nxt nxt。

那么就有 ( c u r − n x t ) × c n t (cur-nxt)\times cnt (cur−nxt)×cnt 的代价将所有的 c u r cur cur 变成 n x t nxt nxt 。

所有相邻元素如果差距过大,考虑将统一下降的部分额外操作出来,保证我们每次循环的开始都是先用一个操作将整体的数做下降。

这样可以保证一次遍历就可以将所有元素都变成 m i n ( a ) min(a) min(a) 。

时间复杂度: O ( n ) O(n) O(n)

代码

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;

typedef long long ll;

void solve() {
    int n, k;
    cin >> n >> k;

    vector<int> a(n);
    for (int i = 0; i < n; ++i) cin >> a[i];

    sort(a.begin(), a.end(), greater<>());
    if (a.back() == a.front()) {
        cout << "0\n";
        return;
    }

    // 一次 k 能够支撑多少元素下降
    ll ans = 0;
    ll pre = 0;
    ll cnt = 0;
    ll cur = -1;
    for (int i = 0; i < n; ++i) {
        ll s = k;
        // 第一个 s 到某个 a[j - 1]
        int j = i;
        while (j < n && pre - cnt * a[j] <= s) {
            cur = a[j];
            pre += a[j];
            cnt += 1;
            j += 1;
        }

        // pre 肯定是一个统一的值
        if (j > i) {
            s -= pre - cnt * a[j - 1];
            // 然后把剩下部分给整了
            ll dec = s / cnt;
            cur = a[j - 1] - dec;
            pre = cur * cnt;
            ans += 1;
        }

        // pre -> a[j - 1]
        if (j < n) {
            ll single_op = k / cnt;
            ll single_decrease = single_op * cnt;
            ll cnt_op = (cur - a[j] - 1) * cnt / single_decrease;
            ans += cnt_op;
            cur = cur - cnt_op * single_op;
            pre = cur * cnt;
        }

        i = j - 1;
    }

    cout << ans << "\n";
}

int main()
{
    ios::sync_with_stdio(false);
    cin.tie(nullptr);

    int T = 1;
    //cin >> T;
    while (T--) {
        solve();
    }

    return 0;
}
相关推荐
yuanmenghao3 分钟前
车载Linux 系统问题定位方法论与实战系列 - 开篇: 为什么需要一套“系统化”的 Linux 问题定位方法
linux·运维·服务器·数据结构·c++·自动驾驶
丶小鱼丶3 分钟前
Java基础之【排序算法】
java·算法
一只小bit5 分钟前
Qt 对话框全方面详解,包含示例与解析
前端·c++·qt·cpp·页面
柏木乃一6 分钟前
基础IO(上)
linux·服务器·c语言·c++·shell
乐迪信息9 分钟前
乐迪信息:AI视频分析技术用于船舶倾斜监控
大数据·网络·人工智能·算法·无人机
知乎的哥廷根数学学派9 分钟前
基于物理约束指数退化与Hertz接触理论的滚动轴承智能退化趋势分析(Pytorch)
开发语言·人工智能·pytorch·python·深度学习·算法·机器学习
CodeByV12 分钟前
【算法题】字符串
数据结构·算法
Zilliz Planet13 分钟前
官宣,Milvus开源语义高亮模型:告别饱和检索,帮RAG、agent剪枝80%上下文
人工智能·算法·机器学习·剪枝·milvus
机器学习之心17 分钟前
用户用电行为分析|MATLAB基于GWO优化的DBSCAN聚类算法
算法·matlab·聚类
古城小栈18 分钟前
Rust 宏 !
算法·rust