[PA2014]Budowa
[PA2014]Budowa
题目大意:
有A和B两名候选人。共有\(n(n\le1000)\)个人参加投票。他们之间形成了一个树结构,树上的结点有两种身份:专家(叶子结点)或领导(非叶子结点)。每位专家都有自己的选择——支持A和B之中的一个;每位领导都有若干个下属(子结点),领导的选择决定于下属中人数较多的那一方,下属的数目保证为奇数,从而不会出现平局状况。最后,根结点的选择即为选举结果。
目前仍有一些专家处于犹豫未决的状态,只要前去游说,就可获得他的支持。每人每天只能选择游说一名专家。A先开始,两人交替进行,直到每位专家都有了确定的选择。请问A是否有策略保证自己赢得选举胜利?
思路:
首先,假设那些犹豫的专家最后两边都不支持,如果此时已经B占优势,那么A永远没有反超的机会了。
否则枚举每个犹豫的专家,令他支持A,剩下两边都不支持,看一下是否能使A胜利,如果能,就说明第一步可以选择这个专家。
枚举复杂度\(\mathcal O(n)\),计算支持者的复杂度\(\mathcal O(n)\),总时间复杂度\(\mathcal O(n^2)\)。可以通过此题。
实际上,本题还有一种\(\mathcal O(n\log n)\)的做法。
一开始判断是否有解的算法同上,但是可以\(\mathcal O(n)\)求出可以选择的专家。
对于一个点\(x\),若子结点中支持A的人数+\(\lceil\frac{\text{犹豫的人数}}2\rceil\ge\)支持B的人数,则说明可以选择该子树内的专家。
由于要排序,所以时间复杂度是\(\mathcal O(n\log n)\)的。
源代码:
\(\mathcal O(n^2)\):
#include<cstdio>
#include<cctype>
#include<vector>
inline int getint() {
register char ch;
register bool neg=false;
while(!isdigit(ch=getchar())) neg|=ch=='-';
register int x=ch^'0';
while(isdigit(ch=getchar())) x=(((x<<2)+x)<<1)+(ch^'0');
return neg?-x:x;
}
const int N=1001;
int c[N],d[N][2],ans[N];
std::vector<int> e[N];
inline void add_edge(const int &u,const int &v) {
e[u].push_back(v);
}
void dfs(const int &x,const int &par) {
//0: xyw, 1: jry
d[x][0]=d[x][1]=0;
if(c[x]<0) d[x][c[x]+2]++;
for(int i=0;i<c[x];i++) {
const int &y=e[x][i];
if(y==par) continue;
dfs(y,x);
if(d[y][1]==d[y][0]) continue;
d[x][d[y][1]>d[y][0]]++;
}
}
int main() {
const int n=getint();
for(register int i=1;i<=n;i++) {
c[i]=getint();
for(register int j=0;j<c[i];j++) {
add_edge(i,getint());
}
}
dfs(1,0);
if(d[1][1]>d[1][0]) {
puts("NIE");
return 0;
}
for(register int i=1;i<=n;i++) {
if(c[i]==0) {
c[i]=-2;
dfs(1,0);
if(d[1][0]>d[1][1]) {
ans[++ans[0]]=i;
}
c[i]=0;
}
}
printf("TAK %d\n",ans[0]);
for(register int i=1;i<=ans[0];i++) {
printf("%d%c",ans[i]," \n"[i==ans[0]]);
}
return 0;
}
\(\mathcal O(n\log n)\):
#include<cstdio>
#include<cctype>
#include<vector>
#include<algorithm>
inline int getint() {
register char ch;
register bool neg=false;
while(!isdigit(ch=getchar())) neg|=ch=='-';
register int x=ch^'0';
while(isdigit(ch=getchar())) x=(((x<<2)+x)<<1)+(ch^'0');
return neg?-x:x;
}
const int N=1001;
int c[N],d[N][3],ans[N];
std::vector<int> e[N];
inline void add_edge(const int &u,const int &v) {
e[u].push_back(v);
}
void dfs(const int &x,const int &par) {
//0: xyw, 1: jry
if(c[x]<=0) d[x][c[x]+2]++;
for(int i=0;i<c[x];i++) {
const int &y=e[x][i];
if(y==par) continue;
dfs(y,x);
if(d[y][1]==d[y][0]) {
d[x][2]++;
} else {
d[x][d[y][1]>d[y][0]]++;
}
}
}
void solve(const int &x,const int &par) {
if(c[x]==0) ans[++ans[0]]=x;
for(int i=0;i<c[x];i++) {
const int &y=e[x][i];
if(y==par) continue;
if(d[y][0]+1-d[y][2]%2==d[y][1]) {
solve(y,x);
}
}
}
int main() {
const int n=getint();
for(register int i=1;i<=n;i++) {
c[i]=getint();
for(register int j=0;j<c[i];j++) {
add_edge(i,getint());
}
}
dfs(1,0);
if(d[1][1]>d[1][0]) {
puts("NIE");
return 0;
}
if(d[1][1]<d[1][0]) {
for(register int i=1;i<=n;i++) {
if(c[i]==0) {
ans[++ans[0]]=i;
}
}
} else {
solve(1,0);
std::sort(&ans[1],&ans[ans[0]]+1);
}
printf("TAK %d\n",ans[0]);
for(register int i=1;i<=ans[0];i++) {
printf("%d%c",ans[i]," \n"[i==ans[0]]);
}
return 0;
}
[PA2014]Budowa的更多相关文章
- bzoj 3722: PA2014 Final Budowa
3722: PA2014 Final Budowa Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 303 Solved: 108[Submit][St ...
- BZOJ 3721: PA2014 Final Bazarek
3721: PA2014 Final Bazarek Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 645 Solved: 261[Submit][ ...
- BZOJ 3709: [PA2014]Bohater
3709: [PA2014]Bohater Time Limit: 5 Sec Memory Limit: 128 MBSec Special JudgeSubmit: 1050 Solved: ...
- bzoj3714: [PA2014]Kuglarz
[PA2014]KuglarzTime Limit: 20 Sec Memory Limit: 128 MBSubmit: 553 Solved: 317[Submit][Status][Discus ...
- 【贪心】bzoj 3709:[PA2014]Bohater
3709: [PA2014]Bohater Time Limit: 5 Sec Memory Limit: 128 MBSec Special JudgeSubmit: 653 Solved: ...
- BZOJ3715: [PA2014]Lustra
3715: [PA2014]Lustra Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 237 Solved: 149[Submit][Status ...
- BZOJ3709: [PA2014]Bohater
3709: [PA2014]Bohater Time Limit: 5 Sec Memory Limit: 128 MBSec Special JudgeSubmit: 339 Solved: ...
- BZOJ3713: [PA2014]Iloczyn
3713: [PA2014]Iloczyn Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 206 Solved: 112[Submit][Status ...
- BZOJ 3713: [PA2014]Iloczyn( 枚举 )
斐波那契数列<10^9的数很少很少...所以直接暴力枚举就行了... ------------------------------------------------------------- ...
随机推荐
- '{}/{}_frames_{:02d}.npy'.format(dataset, train_or_test, i+1)函数
在阅读有关代码的时候,发现一段代码写为: data_frames = np.load(os.path.join(video_root_path, '{}/{}_frames_{:02d}.npy'.f ...
- 分布式监控系统开发【day37】:表结构设计(二)
一.表结构关系图 二.表结构需求讨论 1.主机表(Host) 1.解决了什么问题? 1.如果我不想让它监控了,就有一个开关的东西给它禁掉2.主机存活状态检测间隔 2.代码 class Host(mod ...
- 关于微信登录授权获取unionid的方法
前言:微信登录授权是目前普遍存在于小程序的,还有一种静默授权方式是微信提供的但是不推荐使用,由于不同设备登录openid是不同的那么我们应该怎样拿到一个唯一的ID呢,下面做分享 wxml代码 < ...
- Vim使用技巧:将Tab转换为4个空格
一 Tab转成4个空格 为了防止因为在不同系统中Tab键的宽度不一致而导致代码缩进显示混乱的情况,有必要将Tab键转换成空格,推荐的空格数为4.将下面的代码写入你的.vimrc文件中即可实现在Vim编 ...
- django - 总结 - 用户认证组件
用户认证组件 from django.contrib import auth 从auth_user表中获取对象,没有返回None,其中密码为密文,使用了加密算法 user = auth.authent ...
- [再寄小读者之数学篇](2015-06-24 Series)
(AMM. Problems and Solutions. 2015. 03) Let $\sed{a_n}$ be a monotone decreasing sequence of real nu ...
- 关于JS中的常用表单验证+正则表达式
一.非空验证 trim:去空格(去掉前后的空格),任何字符串都可以用这个方法.写法为:if(v.trim().length==0),表示如果去掉空格后的字符串的长度为0. <body> & ...
- springMVC中controller的几种返回类型
==网文1,还不错,感觉比较老旧springMVC中controller的几种返回类型 - CSDN博客http://blog.csdn.net/qq_16071145/article/details ...
- html 超链接标签 锚点 a标签伪类
一个简易的连接 <a href="01.html">01</a> <body> <a href="01.html" t ...
- Lua中的类型与值
[基础介绍] Lua是一种动态类型的语言.在语言中没有类型定义的语法,每个值都带有其自身的类型信息.在Lua中有8中基本类型,分别是: nil(空)类型 boolean(布尔)类型 number(数字 ...