题目链接:Here

题意:把 \(n\) 个数变成 \(k\) 个相同的数,每次可以把 \(n\) 个数里最大的 \(-1\) 或最小的 \(+1\) ,问最小改变次数

思路:

我们可以枚举,把 \(n\) 个数变成 \(k\) 个 \(a[i]\) (这个相同的数一定是数组里的数,因为如果不是,那么改变次数一定会比正常多)

如果相同的数大于 \(k\) 个,那么改变次数为 \(0\) ,特判掉

有三种情况,一种是只动前面,一种只动后面,还有就是前后都动

因为是改变最大或最小的数,所以我们只有把所有小于 \(a[i]\) 的数变成 \(a[i]-1\) (或者大于 \(a[i]\) 的数变成\(a[i]+1\) )才能进行下一次的改变

然后接着考虑,在什么情况下可以动前面呢,当然是他前面的数大于\((k-1)\)个,同理,在他后面的数大于 \((k-1)\) 个时才可以动后面,然后在任何情况下都可以前后都动( 在$(i=1) $时就相当于是动后面结果不冲突)

以只动前面为例

\(tem1 = (\sum\limits_{j=1}^i((a[i] - 1) -a[j]) + k\)

化简一下发现

\(tem1 = \sum\limits_{j=1}^i(a[i] - 1) -\sum\limits_{j=1}^ia[j] + k\)​

就是 \(i*(a[i]-1)-a[i]\) 的前缀和 \(+k\) ,提前搞一个前缀和可以降低时间复杂度

只动后面同理

\(tmp2 = \sum\limits_{j=i}^na[i] -\sum\limits_{j=i}^n(a[j] + 1) + k\)

动两边,这时相等的数的个数恰好为 \(n\) ,把他们都搞成 \(a[i]\) 然后再减掉多余的

\(tmp3 = \sum\limits_{j=i}^na[j] - \sum\limits_{j=1}^ia[j] + \sum\limits_{j=1}^ia[i] - \sum\limits_{j=i}^na[i] - (n - k)\)

记录好前缀和 和(后缀和?)就可以用 \(\mathcal{O}(n)\)​ 的复杂度解决掉这个问题了

#include<bits/stdc++.h>
using namespace std;
#define ll long long
const ll inf = 1e17;
ll a[200009];
ll cnt[200009];
ll sumq[200009], sumh[200009];
int main() {
int n, k;
ll ans = inf;
scanf("%d%d", &n, &k);
for (int i = 1; i <= n; i++) {
scanf("%lld", &a[i]);
}
sort(a + 1, a + 1 + n);
for (int i = 1; i <= n; i++) sumq[i] = sumq[i - 1] + a[i];
for (int i = n; i >= 1; i--) sumh[i] = sumh[i + 1] + a[i];
for (int i = 1; i <= n; i++) {
if (a[i] == a[i - 1])cnt[i] = cnt[i - 1] + 1;
else cnt[i] = 1;
if (cnt[i] >= k) {
puts("0");
return 0;
}
}
for (int i = 1; i <= n; i++) {
if (i >= k) {
ll tem1 = i * (a[i] - 1) - sumq[i] + k;
ans = min(tem1, ans);
}
if (n - i + 1 >= k) {
ll tem2 = n - i + 1;
tem2 = sumh[i] - tem2 * (a[i] + 1) + k;
ans = min(tem2, ans);
}
if (i < k && (n - i + 1) < k) {
ll tem3 = i * a[i] - sumq[i] + sumh[i] - (n - i + 1) * a[i] - (n - k);
ans = min(tem3, ans);
}
}
printf("%lld\n", ans);
return 0;
}

便捷写法

int main() {
cin.tie(nullptr)->sync_with_stdio(false);
int n, k;
cin >> n >> k;
vector<ll> a(n);
for (ll &x : a) cin >> x;
sort(a.begin(), a.end());
for (int i = 0; i + k - 1 < n; ++i) {
if (a[i] == a[i + k - 1])
return printf("0\n"), 0;
} ll lcost = 0;
ll rcost = 0;
for (int i = 0; i < k; ++i) {
lcost += a[k - 1] - a[i];
rcost += a[n - 1 - i] - a[n - k];
}
for (int j = k; j < n; ++j) {
if (a[k - 1] == a[j]) lcost--;
if (a[n - k] == a[n - 1 - j]) rcost--;
}
ll sum = 0;
for (int i = 0; i < n - 1 - i; ++i) sum += a[n - 1 - i] - a[i];
ll ans = min(sum - (n - k), min(lcost, rcost));
cout << ans;
}

题解 | CF1328F. Make k Equal (思维,前缀 & 后缀和)的更多相关文章

  1. [JZOJ5280]膜法师题解--思维+前缀和

    [JZOJ5280]膜法师题解--思维+前缀和 题目链接 暴 力 过 于

  2. Codeforces Round #320 (Div. 2) [Bayan Thanks-Round] D "Or" Game 枚举+前缀后缀

                                                                            D. "Or" Game       ...

  3. POJ 2752 (KMP 所有可能长度的前缀后缀) Seek the Name, Seek the Fame

    题意: 求一个字符串的相同前缀后缀的所有可能的长度,这里该字符串其本身也算自己的前缀和后缀. 分析: 我们知道next数组的性质是,该字符之前的字符串的最大相同前缀后缀. 既然知道了最大的,即next ...

  4. hdoj 2594 Simpsons’ Hidden Talents 【KMP】【求串的最长公共前缀后缀】

    Simpsons' Hidden Talents Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java ...

  5. CF-1328 F. Make k Equal

    F. Make k Equal 题目链接 题意 长度为n的序列,每次可以选择一个最大的数字将其减一或者选择一个最小的数字将其加一,问最少操作多少次可以使得序列中至少存在 k 个一样的数字 分析 官方题 ...

  6. k倍区间 前缀和【蓝桥杯2017 C/C++ B组】

    标题: k倍区间 给定一个长度为N的数列,A1, A2, ... AN,如果其中一段连续的子序列Ai, Ai+1, ... Aj(i <= j)之和是K的倍数,我们就称这个区间[i, j]是K倍 ...

  7. 1280 前缀后缀集合(map)

    1280 前缀后缀集合 题目来源: Codility 基准时间限制:1 秒 空间限制:131072 KB 分值: 40 难度:4级算法题 一个数组包含N个正整数,其中有些是重复的.一个前缀后缀集是满足 ...

  8. kuangbin专题七 HDU1540 Tunnel Warfare (前缀后缀线段树)

    During the War of Resistance Against Japan, tunnel warfare was carried out extensively in the vast a ...

  9. jQuery EasyUI/TopJUI基本的数字输入框(保留两位小数,带前缀后缀...)

    jQuery EasyUI/TopJUI基本的数字输入框(保留两位小数,带前缀后缀...) numberbox(数值输入框) HTML required:必填字段,默认为false:prompt:显示 ...

  10. 求x!在k进制下后缀零的个数(洛谷月赛T1)

    求x!在k进制下后缀和的个数 20分:     求十进制下的x!后缀和的个数 40分: 高精求阶乘,直接模拟过程 (我不管反正我不打,本蒟蒻最讨厌高精了) 60分     利用一个定理(网上有求x!在 ...

随机推荐

  1. 实例讲解SpringBoot集成Dubbo的步骤及过程

    首先,让我们先了解一下Spring Boot和Dubbo. Spring Boot 是一个开源的 Java Web 框架,它可以帮助开发者快速创建独立的.生产级别的 Spring 应用程序.Sprin ...

  2. CTT2023 邮寄

    从广州被邮寄到了苏州.还有点感冒有点咳嗽,体温 37 度.还是来了. Day 0 清早坐 xp 的车,早上坐飞机,中午坐高铁,下午坐大巴,风尘仆仆地赶到了苏州. 飞机上有一套省选题要验,看了两眼,T1 ...

  3. [ABC264G] String Fair

    Problem Statement In a string fair, they determine the beauty of a non-empty string $S$ consisting o ...

  4. 关于C#接口的用法详细解答,附上案例说明!

    接口 C#中的接口是一种定义了一组方法.属性和事件的类型.它只包含成员的声明,而不包含任何实现.接口可以被类通过实现的方式使用,从而使类能够具有接口定义的行为. 接口在C#中被定义为使用interfa ...

  5. django-celery-results - 使用 Django ORM/Cache 作为结果后端

    https://docs.celeryq.dev/en/stable/django/first-steps-with-django.html#django-celery-results-using-t ...

  6. 动态规划问题(四)最长双音序列长度(LBS)

    问题描述 ​ 以一个乱序的数组,求它的最长双音序列长度.双音序列指该序列先递增,再递减. ​ 如: 对于序列 {1, 11, 2, 10, 4, 5, 2, 1},它的最长双音序列长度为 6, 对于序 ...

  7. 文心一言 VS 讯飞星火 VS chatgpt (38)-- 算法导论5.4 2题

    二.假设我们将球投入到b个箱子里,直到某个箱子中有两个球.每一次投掷都是独立的并且每个球落入任何箱子的机会均等.请问投球次数期望是多少? 文心一言: 这是一个典型的鸽巢原理(Pigeonhole Pr ...

  8. 一种DWS迁移Oracle的CONNECT BY语法的方案

    摘要:本文提供一种GaussDB DWS迁移CONNECT BY语法方案. 本文分享自华为云社区<GaussDB(DWS)迁移 - oracle兼容 -- CONNECT BY迁移>,作者 ...

  9. 如何快速上手 angular.js

    摘要:angular.js 准确的来说,应该不是一个框架,是一个 js 库,一个依赖于 jQuery 的进一步封装,去除繁琐的 DOM 操作,使用数据驱动的 MVC 模块化库. 哎,很难受,连续两个大 ...

  10. CPU高速缓存与极性代码设计

    摘要:CPU内置少量的高速缓存的重要性不言而喻,在体积.成本.效率等因素下产生了当今用到的计算机的存储结构. 介绍 cpu缓存的结构 缓存的存取与一致 代码设计的考量 最后 CPU频率太快,其处理速度 ...