题目链接http://acm.hdu.edu.cn/showproblem.php?pid=5963

吐槽

这道题我第一眼看,嗯??博弈论?还是树上的?我好像不会啊。。。但是一想某人的话,感觉这个应该也不会太难,可能有规律

分析

于是我就从样例开始仔细思考找规律,第一个样例应该是看不出来啥,但第二个内容量就比较丰富了。但我模拟完样例二依旧没发现什么,难道这道题真要建个线段树什么的??接着我把关注点放到了输出上,输出只有两种,那是不是应该存在某种奇偶关系?还是要先考虑链的情况,因为从无根树上的一个点出发遍历就相当于走几条链(好像树的问题大部分都跟链有关)。下面模拟一下



首先要注意到题目中的暗示,每次都要找一个到父亲节点权值为1的点,这就说明了这个问题的限制,它总会结束。先不考虑修改,如果以1为根,不难看出女孩会赢,其中一种走法是







当然也有别的走法,但总会是女生赢。

如果以2为根呢?不难得出还是女孩赢,其中一种走法为







那么是不是图中边权为1的边数为偶数时,就是女生赢呢?仍旧以2为根,我们在4后边再跟一个节点。



会发现这样还是女生赢。

那在根节点2后边跟这个节点呢?



这时候再去模拟就会发现是男生赢了。所以我们猜测,让女生赢的并不是图中边权为1的边数,而是根节点周围边权为1的边数,根据这个大胆的猜测,我写出了下面的代码,好像还白写了一个加边函数。如果你去用样例测试,发现是对的,所以我兴奋的交了上去,WA

#include<cstdio>
#include<cstring>
using namespace std;
const int N=4e4+10;
struct Edge{
int to,nxt,val;
}e[N<<1];
int Head[N],len;
void Ins(int a,int b,int c){
e[++len].to=b;e[len].val=c;
e[len].nxt=Head[a];Head[a]=len;
}
int cnt[N];
int main(){
int t;
scanf("%d",&t);
while(t--){
int n,m;
scanf("%d%d",&n,&m);
memset(cnt,0,sizeof(cnt));
for(int i=1;i<n;i++){
int a,b,c;
scanf("%d%d%d",&a,&b,&c);
cnt[a]+=c;
cnt[b]+=c;
}
for(int i=1;i<=m;i++){
int op,x,y,z;
scanf("%d",&op);
if(op==1){
scanf("%d%d%d",&x,&y,&z);
if(z==1)cnt[x]++,cnt[y]++;
else cnt[x]--,cnt[y]--;
}
else {
scanf("%d",&x);
if(cnt&1)printf("Girls win!\n");
else printf("Boys win!\n");
}
}
}
}

二次分析

既然这个代码过了样例,就说明它应该不是偶然,所以应该是我少考虑了什么。再回去读了一边题,发现题中并没有说当\(op==1\)时,变换的权值与原来的权值相等,也就是说如果我每次都把1变为0,0变为1,那么我上边的代码是可以的,也就是样例情况,但问题就出在它可能是1变为1,0变为0,所以每次必须扫描一边根周围的边权之和,即下边的代码。

#include<cstdio>
#include<cstring>
using namespace std;
const int N=4e4+10;
struct Edge{
int to,nxt,val;
}e[N<<1];
int Head[N],len;
void Ins(int a,int b,int c){
e[++len].to=b;e[len].val=c;
e[len].nxt=Head[a];Head[a]=len;
}
int main(){
int t;
scanf("%d",&t);
while(t--){
int n,m;
scanf("%d%d",&n,&m);
memset(Head,0,sizeof(Head));
len=0;
for(int i=1;i<n;i++){
int a,b,c;
scanf("%d%d%d",&a,&b,&c);
Ins(a,b,c);Ins(b,a,c);
}
for(int i=1;i<=m;i++){
int op,x,y,z;
scanf("%d",&op);
if(op==1){
scanf("%d%d%d",&x,&y,&z);
for(int i=Head[x];i;i=e[i].nxt)
if(e[i].to==y)e[i].val=z;
for(int i=Head[y];i;i=e[i].nxt)
if(e[i].to==x)e[i].val=z;
}
else {
int cnt=0;
scanf("%d",&x);
for(int i=Head[x];i;i=e[i].nxt)cnt+=e[i].val;
if(cnt&1)printf("Girls win!\n");
else printf("Boys win!\n");
}
}
}
}

证明

写完之后感觉就这么过去的话不是很好,万一猜错了就很尴尬,所以想一下证明。

为了简化问题,我们只把链的情况证明了就好。如果连接根节点的边权值为1,其余边有两种情况,一是全为1的,二是含0的,分开讨论一下。

如果含有0,



因为是女生先任意选点,所以让女生选择最后一个点,这样连接根节点的那条边就会被置为0,而因为含有0,所以肯定还会剩下边权为1的,这时由于男生不得不去变换边,所以一定会去变换权值为1的,由于根节点在男生变换的时候为0,所以这样的话只有可能是女生让根节点变为0,也就是只有女生会赢。

如果不含0都是1呢?那女生就可以一次性让男生没的变换,仍然是女生赢。

当连接根节点的边权值为0时就会反过来,证毕。

HDU-5963 朋友 思维的更多相关文章

  1. HDU 5963 朋友 (找规律,思维)

    HDU 5963 朋友 题目大意 B君在围观一群男生和一群女生玩游戏,具体来说游戏是这样的: 给出一棵n个节点的树,这棵树的每条边有一个权值,这个权值只可能是0或1. 在一局游戏开始时,会确定一个节点 ...

  2. HDU - 5963 朋友(思维题)

    题干 B君在围观一群男生和一群女生玩游戏,具体来说游戏是这样的: 给出一棵n个节点的树,这棵树的每条边有一个权值,这个权值只可能是0或1. 在一局游戏开始时,会确定一个节点作为根.接下来从女生开始,双 ...

  3. HDU 5963 朋友 【博弈论】 (2016年中国大学生程序设计竞赛(合肥))

    朋友 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Problem Descr ...

  4. HDU 5963 朋友(找规律博弈)

    http://acm.hdu.edu.cn/showproblem.php?pid=5963 题意: 思路: 我们可以先只考虑单链,自己试几种案例就可以发现规律,只有与根相连的边为1时,只需要奇数次操 ...

  5. hdu 5963:朋友

    刚看到这题时感觉是树上博弈,然后我开始用一维的数据找规律.发现在一维的树上,如果把各边的值合在一起当成一个二进制数,那么,ans只与奇偶性有关,于是,我提出了一个比较大胆的假设:若连接在root上的所 ...

  6. hdu 5963 朋友(2016ccpc 合肥站 C题)

    朋友 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Submissi ...

  7. 【不知道怎么分类】HDU - 5963 朋友

    题目内容 B君在围观一群男生和一群女生玩游戏,具体来说游戏是这样的: 给出一棵n个节点的树,这棵树的每条边有一个权值,这个权值只可能是0或1. 在一局游戏开始时,会确定一个节点作为根.接下来从女生开始 ...

  8. HDU 5963 朋友(树+博弈)

    #include<vector> #include<cstdio> #include<cstring> #include<algorithm> #def ...

  9. 【博弈】HDU - 5963 朋友

    题目 B君在围观一群男生和一群女生玩游戏,具体来说游戏是这样的: 给出一棵n个节点的树,这棵树的每条边有一个权值,这个权值只可能是0或1. 在一局游戏开始时,会确定一个节点作为根.接下来从女生开始,双 ...

  10. 最简单的博弈论——HDU - 5963 朋友 (博弈)

    OK,好的先看一下题意: B君在围观一群男生和一群女生玩游戏,具体来说游戏是这样的: 给出一棵n个节点的树,这棵树的每条边有一个权值,这个权值只可能是0或1. 在一局游戏开始时,会确定一个节点作为根. ...

随机推荐

  1. 异步编程RxJava-介绍

    前言前段时间写了一篇对协程的一些理解,里面提到了不管是协程还是callback,本质上其实提供的是一种异步无阻塞的编程模式:并且介绍了java中对异步无阻赛这种编程模式的支持,主要提到了Future和 ...

  2. JS排序算法--冒泡排序和选择排序

    在我们JS语法当中,数据类型中的复杂数据类型,有一项我们常用的数组数据类型,其中存储的数据有时是乱序的,需要排序,我们有多种方法,最简单的肯定是 :变量.sort(fonction(a,b){a> ...

  3. 039.集群网络-Pod和SVC网络实践

    一 Pod和SVC网络 1.1 实践准备及原理 Docker实现了不同的网络模式,Kubernetes也以一种不同的方式来解决这些网络模式的挑战.本完整实验深入剖析Kubernetes在网络层是如何实 ...

  4. 面试官:说说你对css效率的理解

    大家好,我是小雨小雨,致力于分享有趣的.实用的技术文章. 内容分为翻译和原创,如果有问题,欢迎随时评论或私信,希望和大家一起进步. 大家的支持是我创作的动力. 选择器的优先级 众所周知,选择器是有权重 ...

  5. Linux下安装MySQL的tar.gz包

    以root用户登录待安装的服务器. 上传软件包并解压. 以root用户通过sftp/ftp工具上传“mysql-5.7.24-linux-glibc2.12-x86_64.tar.gz”软件包到“/o ...

  6. Trie 字典树,hdu1251

    参考博客:https://www.cnblogs.com/TheRoadToTheGold/p/6290732.html 字典树就是单词树,顺着一条路径到达终止结点就形成一个单词,该单词的前缀包含在这 ...

  7. vue2源码分析:patch函数

    目录 1.patch函数的脉络 2.类vnode的设计 3.createPatch函数中的辅助函数和patch函数 4.源码运行展示(DEMO) 一.patch函数的脉络 首先梳理一下patch函数的 ...

  8. sql-lib闯关21-30

    第二十一关 base64编码,单引号,报错型,cookie型注入. 本关和less-20相似,只是cookie的uname值经过base64编码了,下图为我们输入万能密码显示 uname = YWRt ...

  9. css 重排与重绘

    css 重绘与重排 我们要知道当浏览器下载完页面的所有资源后,就会开始解析源代码. HTML 会被解析成 DOM Tree,Css 则会被渲染成 CSSOM Tree,最后它们会附加到一起,形成渲染树 ...

  10. 使用PyTorch进行情侣幸福度测试指南

    欢迎关注磐创博客资源汇总站: http://docs.panchuang.net/ 欢迎关注PyTorch官方中文教程站: http://pytorch.panchuang.net/ 计算机视觉–图像 ...