题目链接: http://codeforces.com/problemset/problem/657/C

--------------------------------------------------------------------------------------------------------

题目的特别之处在于只有 $+1$ $+5$ 这两种操作 我们要考虑如何利用这个条件

多想一下后可以发现 如果最优解的目标值为$x($将至少$k$个人的值增加到$x)$

那么一定存在一个人 他的初始值在 $[x - 4, x]$ 这个范围内

否则将$x$减去$5$后可以得到更优的解

因此可能成为最优解的目标值最多只有 $n * 5$ 种

现在考虑的便是在 $O(n)$ 枚举目标值的前提下 如何对于每个目标值快速计算出答案

我一开始的想法是根据 $mod 5$ 的余数分类 写$5$个数组记录下前缀和什么的

然而这样二分答案有一个 $log$ 二分数组下标又一个 $log$

尽管题目给了 $4s$ 应该可以过 但是总感觉这样做不够优雅

再次分析题目条件我们有可以发现 实际上每次询问都是与目标距离最小的 $k$ 个数 而这个$k$是不变的

于是现在问题就变成了维护一个数据结构 支持查找容器内最小的 $k$个数 以及添加一个数

这显然就是一个堆了 总的复杂度是 $O(n + nlogk)$

由于在 $mod 5$ 的$5$中情况下 所有初始值转移到目标值的代价多少的排序并不是一样的

因此我们维护$5$个堆就好

 #include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <queue>
using namespace std;
const int N = 2e5 + ;
const long long inf = 1e9;
int a[N], dis[N * ];
long long sum[];
long long ans = 1e18;
int n, k, b, c, cnt;
priority_queue <long long> q[];
int main()
{
scanf("%d%d%d%d", &n, &k, &b, &c);
b = min(b, c * );
for(int i = ; i <= n; ++i)
{
scanf("%d", &a[i]);
for(int j = ; j < ; ++j)
dis[cnt++] = a[i] + j;
}
sort(a + , a + + n);
sort(dis, dis + cnt);
cnt = unique(dis, dis + cnt) - dis;
int now = ;
for(int i = ; i < cnt; ++i)
{
int x = dis[i], kind =(inf + x) % ;
long long cost;
while(now <= n && a[now] <= x)
{
for(int j = ; j < ; ++j)
{
cost = (inf + j - a[now]) / * b + (inf + j - a[now]) % * c;
q[j].push(cost);
sum[j] += cost;
if((int)q[j].size() > k)
{
sum[j] -= q[j].top();
q[j].pop();
}
}
++now;
}
long long y = (inf + kind - x) / * b * k;
if((int)q[kind].size() == k)
ans = min(ans, sum[kind] - y);
}
printf("%lld\n", ans);
return ;
}

codeforces 657C - Bear and Contribution [想法题]的更多相关文章

  1. Codeforces 639D Bear and Contribution

    Bear and Contribution 对于对于5余数为, 0, 1, 2, 3, 4的分别处理一次, 用优先队列贪心. #include<bits/stdc++.h> #define ...

  2. CodeForces - 156B Suspects 逻辑 线性 想法 题

    题意:有1~N,n(1e5)个嫌疑人,有m个人说真话,每个人的陈述都形如X是凶手,或X不是凶手.现在给出n,m及n个陈述(以+x/-X表示)要求输出每个人说的话是true ,false or notd ...

  3. CodeForces - 798D Mike and distribution 想法题,数学证明

    题意:给你两个数列a,b,你要输出k个下标,使得这些下标对应的a的和大于整个a数列的和的1/2.同时这些下标对应的b //题解:首先将条件换一种说法,就是要取floor(n/2)+1个数使得这些数大于 ...

  4. CodeForces - 55C Pie or die 想法题(猜程序)

    http://codeforces.com/problemset/problem/55/C 题意:一个博弈. 题解:瞎猜,目前不清楚原理 #include<iostream> #inclu ...

  5. codeforces 584E Anton and Ira [想法题]

    题意简述: 给定一个$1$到$n(n<=2000)$的初始排列以及最终排列 我们每次可以选取位置为$i$和$j$的 并交换它们的位置 花费为$ |i-j| $ 求从初始状态变换到末状态所需最小花 ...

  6. codeforces gym 100345I Segment Transformations [想法题]

    题意简述 给定一个由A C G T四个字母组成的密码锁(每拨动一次 A变C C变G G变T T变A) 密码锁有n位 规定每次操作可以选取连续的一段拨动1~3次 问最少几次操作可以将初始状态变到末状态 ...

  7. codeforces 11 B.Jumping Jack 想法题

    B. Jumping Jack Jack is working on his jumping skills recently. Currently he's located at point zero ...

  8. CodeForces 111B - Petya and Divisors 统计..想法题

    找每个数的约数(暴力就够了...1~x^0.5)....看这约数的倍数最后是哪个数...若距离大于了y..统计++...然后将这个约数的最后倍数赋值为当前位置...好叼的想法题.... Program ...

  9. Codeforces 385C Bear and Prime Numbers

    题目链接:Codeforces 385C Bear and Prime Numbers 这题告诉我仅仅有询问没有更新通常是不用线段树的.或者说还有比线段树更简单的方法. 用一个sum数组记录前n项和, ...

随机推荐

  1. C++中操作符重载的概念

    1,下面的复数解决方案是否可行? 1,代码示例: class Comples { public: int a; int b; }; int main() { Complex c1 = {, }; Co ...

  2. 51nod1769 Clarke and math 2

    题目 实际上就是要求\(f*I^k\). 因为\(I^k\)是一个积性函数,所以我们只需要考虑如何求\(I^k(p^a)\). 把这个东西转化成一个长度为\(k\)的序列,每一位是\(\frac{i_ ...

  3. 剑指offer-二维数组中的查找-数组-python

    题目描述 在一个二维数组中(每个一维数组的长度相同),每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数 ...

  4. wordpress添加自定义菜单栏

    1. 以下代码是wordpress的wp_nav_menu(); <?php $defaults = array( 'container'=>false, //父级div为空 'link_ ...

  5. python-xss攻击和单例模式

    1.python中单例模式 class Foo: instance = None def __new__(cls, *args, **kwargs): if not Foo.instance: Foo ...

  6. Oracle锁表信息处理步骤

    查看是否有锁表的sql select 'blocker(' || lb.sid || ':' || sb.username || ')-sql:' || qb.sql_text blockers, ' ...

  7. nginx之热部署,以及版本回滚

    热部署的概念:当从老版本替换为新版本的nginx的时候,如果不热部署的话,会需要取消nginx服务并重启服务才能替换成功,这样的话会使正在访问的用户在断开连接,所以为了不影响用户的体验,且需要版本升级 ...

  8. C 调试 gdb常用命令

    gdb常用命令: [root@redhat home]#gdb 调试文件:启动gdb (gdb) l :(字母l)从第一行开始列出源码 (gdb) break n :在第n行处设置断点 (gdb) b ...

  9. LVS集群的ipvsadm命令用法

    准备一台Linux服务器,安装ipvsadm软件包,练习使用ipvsadm命令,实现如下功能: - 使用命令添加基于TCP一些的集群服务 - 在集群中添加若干台后端真实服务器 - 实现同一客户端访问, ...

  10. Linux 安装 Composer

    Linux 安装 Composer  入门 练习环境: 虚拟机:Oracle VM VirtualBox. 系统:CentOS 7. 安装方式一: 参考网址:https://learnku.com/c ...