题目传送门

题目大意

给出一个 \(n\) 个点 \(m\) 条无向边的图,每个点都有一个 \(\in [0,1]\) 的权值,每次可以选择一条边,然后将该边相连两点权值异或上 \(1\)。问有多少种选择方法使得每个点的权值都变为 \(0\)。(每条边只能选择一次)

但是这个问题太简单了,所以你要求删掉每个点以及它连出的边之后的答案。

有 \(t\) 组数据,\(t\le 5,n,m\le 10^5\)。

思路

这道题不是很难写,但是很难想。

我们先考虑一棵树的答案,你发现这样方案是唯一的,因为你从叶子节点开始考虑,如果当前点是黑的,那就选择父亲边,否则就不能选。然后你发现有解的充要条件就是黑点是偶数个。

考虑一个 \(n\) 点 \(m\) 边的连通图,你发现如果我们钦点出 \(n-1\) 条边作为树边,那么它们是惟一的。对于剩下的 \(m-n+1\) 条边,你可以任意选择要或不要(相当于改变两个点的初始颜色),然后改变那 \(n-1\) 条边使得合法即可。所以答案就是 \(2^{m-n+1}\) ,判断有解的方法同上。

对于一个不连通图,你发现如果有解的话答案就是 \(2^{m-n+t}\)。其中 \(t\) 是连通块个数。显然。

考虑删掉一个点以及相连的边后的方案数,实际上就是如何判断合不合法以及删掉之后的连通块个数。

  • 连通块个数

你发现这个东西只跟割边有关,设 \(\text{cnt}(i)\) 表示点 \(i\) 在 \(\text{dfs}\) 树上与儿子节点相连的割边个数,你发现答案连通块个数就增加了 \(\text{cnt}(i)\) 个。(如果 \(i=\text{root}\) 的话 \(\text{cnt}(i)\) 需要减 \(1\) 因为它连通块增加个数会少一个,因为它没有父亲)

  • 判断合法性

首先需要判断除了该点所属外的连通块的合法性。

然后判断分出来的子树合法性。

然后判断剩下的一块合法性就好了。

合法性判断同上文,看连通块里面 \(1\) 的个数就好了。


用 \(\text{Tarjan}\) 实现,时间复杂度 \(\Theta(Tn)\)。

\(\texttt{Code}\)

#include <bits/stdc++.h>
using namespace std; #define Int register int
#define mod 1000000007
#define MAXN 100005 template <typename T> inline void read (T &t){t = 0;char c = getchar();int f = 1;while (c < '0' || c > '9'){if (c == '-') f = -f;c = getchar();}while (c >= '0' && c <= '9'){t = (t << 3) + (t << 1) + c - '0';c = getchar();} t *= f;}
template <typename T,typename ... Args> inline void read (T &t,Args&... args){read (t);read (args...);}
template <typename T> inline void write (T x){if (x < 0){x = -x;putchar ('-');}if (x > 9) write (x / 10);putchar (x % 10 + '0');} vector <int> G[MAXN];
int n,m,rt,ind,a[MAXN],f[MAXN],d[MAXN],bin[MAXN],dfn[MAXN],low[MAXN],deg[MAXN],cnt[MAXN],bel[MAXN],sum[MAXN]; void Link (int x,int y){
deg[x] ++,deg[y] ++;
G[x].push_back (y),G[y].push_back (x);
} void Tarjan (int u){
bel[u] = rt,dfn[u] = low[u] = ++ ind,sum[u] = a[u];
for (Int v : G[u]){
if (!dfn[v]){
Tarjan (v);
sum[u] += sum[v];
low[u] = min (low[u],low[v]);
if (low[v] >= dfn[u]) cnt[u] ++,f[u] += sum[v],d[u] |= (sum[v] & 1);//u->v是割边
}
else low[u] = min (low[u],dfn[v]);
}
cnt[u] -= (u == rt);
} signed main(){
int t;read (t);
bin[0] = 1;for (Int i = 1;i <= MAXN - 5;++ i) bin[i] = 2 * bin[i - 1] % mod;
while (t --> 0){
read (n,m);int val = 0,tot = 0;
for (Int i = 1,u,v;i <= m;++ i) read (u,v),Link (u,v);
for (Int i = 1;i <= n;++ i) scanf ("%1d",&a[i]);
for (Int i = 1;i <= n;++ i) if (!dfn[i]) rt = i,Tarjan (i),tot ++,val += (sum[i] & 1);
write (val ? 0 : bin[m - n + tot]);
for (Int i = 1;i <= n;++ i){
putchar (' ');
if (d[i]) putchar ('0');
else if (val - (sum[bel[i]] & 1)) putchar ('0');
else if ((sum[bel[i]] - f[i] - a[i]) & 1) putchar ('0');
else write (bin[(m - deg[i]) - (n - 1) + tot + cnt[i]]);
}
putchar ('\n'),ind = 0;
for (Int i = 1;i <= n;++ i) G[i].clear (),f[i] = d[i] = cnt[i] = deg[i] = dfn[i] = low[i] = 0;
}
return 0;
}

题解 [HAOI2018]反色游戏的更多相关文章

  1. 【BZOJ5303】[HAOI2018]反色游戏(Tarjan,线性基)

    [BZOJ5303][HAOI2018]反色游戏(Tarjan,线性基) 题面 BZOJ 洛谷 题解 把所有点全部看成一个\(01\)串,那么每次选择一条边意味着在这个\(01\)串的基础上异或上一个 ...

  2. bzoj 5393 [HAOI2018] 反色游戏

    bzoj 5393 [HAOI2018] 反色游戏 Link Solution 最简单的性质:如果一个连通块黑点个数是奇数个,那么就是零(每次只能改变 \(0/2\) 个黑点) 所以我们只考虑偶数个黑 ...

  3. P4494 [HAOI2018]反色游戏

    P4494 [HAOI2018]反色游戏 题意 给你一个无向图,图上每个点是黑色或者白色.你可以将一条边的两个端点颜色取反.问你有多少种方法每个边至多取反一次使得图上全变成白色的点. 思路 若任意一个 ...

  4. 【loj#2524】【bzoj5303】 [Haoi2018]反色游戏(圆方树)

    题目传送门:loj bzoj 题意中的游戏方案可以转化为一个异或方程组的解,将边作为变量,点作为方程,因此若方程有解,方程的解的方案数就是2的自由元个数次方.我们观察一下方程,就可以发现自由元数量=边 ...

  5. [BZOJ5303] [HAOI2018] 反色游戏

    题目链接 LOJ:https://loj.ac/problem/2524 BZOJ:https://lydsy.com/JudgeOnline/problem.php?id=5303 洛谷:https ...

  6. [BZOJ5303][HAOI2018]反色游戏(Tarjan)

    暴力做法是列异或方程组后高斯消元,答案为2^自由元个数,可以得60分.但这个算法已经到此为止了. 从图论的角度考虑这个问题,当原图是一棵树时,可以从叶子开始唯一确定每条边的选择情况,所以答案为1. 于 ...

  7. 洛谷P4494 [HAOI2018]反色游戏(tarjan)

    题面 传送门 题解 我们先来考虑一个联通块,这些关系显然可以写成一个异或方程组的形式,形如\(\oplus_{e\in edge_u}x_e=col_u\) 如果这个联通块的黑色点个数为奇数,那么显然 ...

  8. [HAOI2018]反色游戏

    [Luogu4494] [BZOJ5303] [LOJ2524] LOJ有数据就是好 原题解,主要是代码参考 对于每一个联通块(n个点),其他的边一开始随便选,只需要n-1条边就可以确定最终结果. 所 ...

  9. bzoj 5303: [Haoi2018]反色游戏

    Description Solution 对于一个有偶数个黑点的连通块,只需要任意两两配对,并把配对点上的任一条路径取反,就可以变成全白了 如果存在奇数个黑点的连通块显然无解,判掉就可以了 如果有解, ...

随机推荐

  1. tensorflow 单机多卡 官方cifar10例程

    测试了官方历程,看没有问题,加上时间紧任务重,就不深究了. 官方tutorials:https://www.tensorflow.org/tutorials/images/deep_cnn githu ...

  2. 百闻不如一试——公式图片转Latex代码

    写博客时,数学公式的编辑比较占用时间,在上一篇中详细介绍了如何在Markdown中编辑数学符号与公式. https://www.cnblogs.com/bytesfly/p/markdown-form ...

  3. RHEL7.2系统下的软件管理(yum)、本地yum源和网络yum源的搭建

    在Liunx系统中,rpm和yum都可以安装软件,但rpm存在安装软件的依赖性,yum安装软件需要yum源 1.yum yum install softwarename ##安装 yum repoli ...

  4. Swift-Button 的 highlighted(高亮)

    摘要 在学习小程序时,看到小程序中的一个样式属性 hover-class,通过设置这个属性,就可以给点击的控件添加一个高亮效果.所以也就萌生了在 Swift 也实现一个类似的功能的想法,开干. 下面代 ...

  5. 记一次 .NET 某机械臂智能机器人控制系统MRS CPU爆高分析

    一:背景 1. 讲故事 这是6月中旬一位朋友加wx求助dump的故事,他的程序 cpu爆高UI卡死,问如何解决,截图如下: 在拿到这个dump后,我发现这是一个关于机械臂的MRS程序,哈哈,在机械臂这 ...

  6. Spring系列之不同数据库异常如何抽象的?

    前言 使用Spring-Jdbc的情况下,在有些场景中,我们需要根据数据库报的异常类型的不同,来编写我们的业务代码.比如说,我们有这样一段逻辑,如果我们新插入的记录,存在唯一约束冲突,就会返回给客户端 ...

  7. 【Azure 应用服务】Python flask 应用部署在Aure App Service中作为一个子项目时,解决遇见的404 Not Found问题

    问题描述 在成功的部署Python flask应用到App Service (Windows)后,如果需要把当前项目(如:hiflask)作为一个子项目(子站点),把web.config文件从wwwr ...

  8. Defence

      emm...这道题我调了一下午你敢信??   好吧还是我太天真了.   开始的时候以为自己线段树动态开点与合并写错了,就调;   结果发现没问题,那就是信息维护错了.   一开始以为自己最左右的1 ...

  9. Identity用户管理入门二(显示用户列表)

    在Controllers中新建AccountController,并在构造方法(函数)中注入SignInManager,UserManager UserManager   用户管理(注册,查找,修改, ...

  10. js设计模式之发布订阅模式

    1. 定义 发布-订阅模式其实是一种对象间一对多的依赖关系,当一个对象的状态发送改变时,所有依赖于它的对象都将得到状态改变的通知. 订阅者(Subscriber)把自己想订阅的事件注册(Subscri ...