http://codeforces.com/contest/723/problem/C

题目是给出一个序列

a[i]表示第i个歌曲是第a[i]个人演唱,现在选出前m个人,记b[j]表示第j个人演唱歌曲的数量,

现在你可以改变a[]这个数组,使得前m个人的演唱歌曲的数量的最小值最大。

很明显对于前m个人,每个人唱了多少首是很容易统计出来的,只需要对<=m的人进行统计即可,不用统计>m的,这样的话数组只需开到2000即可。

对于后面的人,可以改变,优先改变前m个人中演唱歌曲最小的那个,这个可以用优先队列维护。

然后如果前m个人本来分配不均匀,那么最小值的最大值得不到最大化。

要对前m个人均匀一下,这个我用了两个优先队列去做,因为思路就是把最大的分一次给最小的,这样的最优的。

hack点:注意到ans最多也只是n / m。就相当于把n分成m分,如果前m个人的答案已经是n / m,后面的人就不需要变化的,不然的话改变数量变大了,会错误。好像wa7就是这个坑。

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
using namespace std;
#define inf (0x3f3f3f3f)
typedef long long int LL; #include <iostream>
#include <sstream>
#include <vector>
#include <set>
#include <map>
#include <queue>
#include <string>
int n, m;
const int maxn = + ;
int book[maxn];
int a[maxn];
struct node {
int num;
int id;
bool operator < (const struct node &rhs) const {
return num > rhs.num;
}
node(int aa, int bb) : num(aa), id(bb) {}
};
struct GGnode {
int num;
int id;
bool operator < (const struct GGnode &rhs) const {
return num < rhs.num;
}
GGnode(int aa, int bb) : num(aa), id(bb) {}
};
priority_queue<struct node>que;
priority_queue<struct GGnode>GGque;
vector<int>pos[maxn];
void work() {
scanf("%d%d", &n, &m);
for (int i = ; i <= n; ++i) {
scanf("%d", &a[i]);
if (a[i] <= m) {
book[a[i]]++;
}
}
for (int i = ; i <= m; ++i) {
que.push(node(book[i], i));
}
int tochange = ;
int get = n / m;
for (int i = ; i <= n; ++i) {
if (a[i] <= m) continue;
struct node t = que.top();
if (t.num == get) break;
que.pop();
a[i] = t.id;
t.num++;
que.push(t);
tochange++;
}
memset(book, , sizeof book);
while (!que.empty()) {
struct node t = que.top();
que.pop();
book[t.id] = t.num;
}
int all = ;
int ans = inf;
for (int i = ; i <= m; ++i) {
ans = min(ans, book[i]);
all += book[i];
que.push(node(book[i], i));
GGque.push(GGnode(book[i], i));
}
for (int i = ; i <= n; ++i) {
if (a[i] <= m) {
pos[a[i]].push_back(i);
}
}
int gold = all / m;
if (ans != gold) {
while (true) {
struct node tt = que.top();
que.pop();
struct GGnode GGtt = GGque.top();
GGque.pop();
if (book[tt.id] == gold) break; book[tt.id]++;
book[GGtt.id]--;
int tpos = pos[GGtt.id].back();
pos[GGtt.id].pop_back();
a[tpos] = tt.id; tt.num = book[tt.id];
GGtt.num = book[GGtt.id]; que.push(tt);
GGque.push(GGtt);
tochange++;
}
}
ans = gold;
cout << ans << " " << tochange << endl;
for (int i = ; i <= n; ++i) {
printf("%d ", a[i]);
}
return;
} int main() {
#ifdef local
freopen("data.txt","r",stdin);
#endif
work();
return ;
}

感觉写的有点麻烦,

所以重写了一次。

考虑到它一定要使得b[j]数组中的最小值最大,而这个最大值,必然是n / m的。所以,直接把1---m中数量小于n / m的统计起来,用其他去变即可。

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
using namespace std;
#define inf (0x3f3f3f3f)
typedef long long int LL; #include <iostream>
#include <sstream>
#include <vector>
#include <set>
#include <map>
#include <queue>
#include <string>
const int maxn = + ;
int a[maxn];
int book[maxn];
set<int>pos;
void work() {
int n, m;
scanf("%d%d", &n, &m);
int gold = n / m;
for (int i = ; i <= n; ++i) {
scanf("%d", &a[i]);
if (a[i] <= m) {
book[a[i]]++;
}
}
for (int i = ; i <= n; ++i) {
if (a[i] <= m && book[a[i]] < gold) {
pos.insert(a[i]);
}
}
for (int i = ; i <= m; ++i) {
if (book[i] < gold) pos.insert(i);
}
// printf("ff");
int toChange = ;
set<int> :: iterator it = pos.begin();
if (it != pos.end()) { //如果需要调整
int val = *it;
for (int i = ; i <= n; ++i) {
if (a[i] > m) {
book[val]++;
a[i] = val;
toChange++;
} else {
if (pos.find(a[i]) != pos.end()) continue;
if (book[a[i]] == gold) continue;
book[a[i]]--;
book[val]++;
a[i] = val;
toChange++;
}
if (book[val] == gold) {
it++;
if (it == pos.end()) break;
val = *it;
}
}
}
printf("%d %d\n", gold, toChange);
for (int i = ; i <= n; ++i) {
printf("%d ", a[i]);
}
return ;
} int main() {
#ifdef local
freopen("data.txt","r",stdin);
#endif
work();
return ;
}

Codeforces Round #375 (Div. 2) Polycarp at the Radio 优先队列模拟题 + 贪心的更多相关文章

  1. Codeforces Round #375 (Div. 2)【A,B【模拟】,D【DFS】】

    PS_B:阿洗吧!B题卧槽数组开了250... PS_D:D题主要挂在了50*50口算得了250,数组开小,然后一开始还错了.= =哎,以后对于数据范围还是注意一点: 卧槽,这场可真二百五了... A ...

  2. Codeforces Round #234 (Div. 2) A. Inna and Choose Options 模拟题

    A. Inna and Choose Options time limit per test 1 second memory limit per test 256 megabytes input st ...

  3. Codeforces Round #368 (Div. 2) A. Brain's Photos (水题)

    Brain's Photos 题目链接: http://codeforces.com/contest/707/problem/A Description Small, but very brave, ...

  4. Codeforces Round #375 (Div. 2) C. Polycarp at the Radio 贪心

    C. Polycarp at the Radio time limit per test 2 seconds memory limit per test 256 megabytes input sta ...

  5. Codeforces Round #375 (Div. 2)

    A. The New Year: Meeting Friends 水 #include <set> #include <map> #include <stack> ...

  6. Codeforces Round #375 (Div. 2) A B C 水 模拟 贪心

    A. The New Year: Meeting Friends time limit per test 1 second memory limit per test 256 megabytes in ...

  7. Codeforces Round #375 (Div. 2) ABCDE

    A - The New Year: Meeting Friends 水 #include<iostream> #include<algorithm> using namespa ...

  8. Codeforces Round #281 (Div. 2) A. Vasya and Football 模拟

    A. Vasya and Football 题目连接: http://codeforces.com/contest/493/problem/A Description Vasya has starte ...

  9. Codeforces Round #375 (Div. 2) - D

    题目链接:http://codeforces.com/contest/723/problem/D 题意:给定n*m小大的字符矩阵.'*'表示陆地,'.'表示水域.然后湖的定义是:如果水域完全被陆地包围 ...

随机推荐

  1. Image Sub-pixel interpolation by Verilog

    bilinear interpolation -------------------------------------------------------- input a0  a1 b0  b1 ...

  2. POJ 1042 Gone Fishing( DP )

    题意:小明打算做一个h((1 <= h <= 16))个小时钓鱼旅行.发现这里有n(2 <= n <= 25)个湖,而且所有的湖都在一条路的旁边.小明打算从第1个湖开始钓起,每 ...

  3. Poj 1125 Stockbroker Grapevine(Floyd算法求结点对的最短路径问题)

    一.Description Stockbrokers are known to overreact to rumours. You have been contracted to develop a ...

  4. 统计不同的单词(map应用)

    题目描述: 输入一些单词,找出所有满足如下条件的单词:该单词不能通过字母重排,得到输入文本中的另一个单词.在判断是否满足条件时,字母不区分大小写,但在输出时应保留输入中的大小写,按字典序进行排列(所有 ...

  5. nginx利用proxy_cache来缓存文件

    为什么要做web cache,我想大家最主要的是解决流量的压力.随着网站流量的提升,如果只是单台机器既处理静态文件,又处理动态脚本,显然效率很难上升,不能处理日益上涨的流量压力.与此同时某些网站的页面 ...

  6. openstack常见问题汇总

    汇总下常见的问题以及解释下一些比较容易让人萌的参数配置等等 问题汇总1.使用纯文本模式进行复制粘贴,打死不要用word!!!可以解决绝大多数问题,如果你依然执迷不悟,那么就好自为之吧 2.创建路由器时 ...

  7. winfrom实现控件全屏效果

    用常规方法实现全屏显示时,由于采用的三方控件导致界面顶端一直有一条半透明的类似标题栏的东西无法去除,原因一直没找到. 下面综合整理下网上两位博主的用WindowsAPI实现全屏的方法: 控件全屏显示: ...

  8. C#设计模式(11)——外观模式

    一.概念 外观模式提供了一个统一的接口,用来访问子系统中的一群接口.外观定义了一个高层接口,让子系统更容易使用.使用外观模式时,我们创建了一个统一的类,用来包装子系统中一个或多个复杂的类,客户端可以直 ...

  9. HTTP之cookie技术

    Cookie由变量名和值组成, 其属性中既有标准的Cookie变量, 也有用户自己创建的变量,属性中变量是用"变量=值"形式来保存 Cookie格式如下: Set-Cookie: ...

  10. [hdu3549]Flow Problem(最大流模板题)

    解题关键:使用的挑战程序设计竞赛上的模板,第一道网络流题目,效率比较低,且用不习惯的vector来建图. 看到网上其他人说此题有重边,需要注意下,此问题只在邻接矩阵建图时会出问题,邻接表不会存在的,也 ...