首先发现对于一个联通块有奇数个黑点,那么总体来说答案无解。这个很容易想,因为对每个边进行操作会同时改变两个点的颜色,异或值不变。

然后一个朴素的想法是写出异或方程进行高斯消元。

可以发现高斯消元的过程实际上就是合并两个点的过程,如果是一棵树的话那么答案一定是2。

对于树上每多的一条边,它在合并点的过程中会被消除掉,这意味着这个是一个自由元。所以我们发现连通图的答案是$2^{m-n+1}$。

这样第一问答案就可以求了。判完无解后答案为$2^{m-n+d}$,$d$为联通块数。

对于第二问,如果有超过两个联通块的黑点为奇数个,后面全部为0。

现在考虑删除一个点,没有改变连通性,那么只有两种情况,一种是其它联通块仍然有奇数个黑点,答案为0,否则看自己这个联通块整体的黑点个数异或上当前删除的点的颜色,为奇数则答案为0,否则答案不难想。

如果删除了一个点,改变了连通性,那么分别检查每个联通块的奇偶性,这个不难,答案也不难想。

代码:

 #include<bits/stdc++.h>
using namespace std; const int maxn = ;
const int mod = 1e9+; int n,m,cnt;
vector<int> g[maxn];
int a[maxn],flag; int low[maxn],dfn[maxn],sz[maxn],cl,pa[maxn];
int ans[maxn],pw2[maxn*]; void init(){
memset(low,,sizeof(low));
memset(dfn,,sizeof(dfn));
memset(sz,,sizeof(sz));
memset(pa,,sizeof(pa));
memset(ans,,sizeof(ans));
memset(pw2,,sizeof(pw2));
memset(a,,sizeof(a));
for(int i=;i<=n;i++) g[i].clear();
n = m = cnt = flag = ;
} void dfs(int now,int fa){
low[now] = dfn[now] = ++cl;
pa[now] = fa; sz[now] = a[now];
for(int i=;i<g[now].size();i++){
int z = g[now][i];
if(z == fa || dfn[z] > dfn[now]) continue;
if(dfn[z] == ){
dfs(z,now); sz[now]^=sz[z];
low[now] = min(low[now],low[z]);
}else low[now] = min(low[now],dfn[z]); }
} void dfs2(int now,int fa,int dr){
for(int i=;i<g[now].size();i++){
if(pa[g[now][i]] != now) continue;
dfs2(g[now][i],now,dr);
}
int isfuck = ;
if(flag - dr) {ans[now] = ;return;}
if(fa == ){
int mlgb = ;
for(int i=;i<g[now].size();i++){
if(pa[g[now][i]] != now) continue;
mlgb++; isfuck += sz[g[now][i]];
}
if(mlgb == ){ans[now] = ans[];}
else if(mlgb == ){
if(isfuck) ans[now] = ;
else ans[now] = pw2[m-g[now].size()-(n-)+cnt];
}else{
if(isfuck) ans[now] = ;
else ans[now] = pw2[m-g[now].size()-(n-)+(cnt-+mlgb)];
}
}else{
int mlgb = ;
for(int i=;i<g[now].size();i++){
if(pa[g[now][i]] != now) continue;
if(low[g[now][i]] >= dfn[now]){
mlgb++;
isfuck += sz[g[now][i]];
}
}
if(mlgb == ){
if(dr ^ a[now]) ans[now] = ;
else ans[now] = pw2[m-g[now].size()-(n-)+cnt];
}else{
if(isfuck) ans[now] = ;
else if(dr^(isfuck&)^a[now]) ans[now] = ;
else ans[now] = pw2[m-g[now].size()-(n-)+(cnt+mlgb)];
}
}
} void read(){
scanf("%d%d",&n,&m);
for(int i=;i<=m;i++){
int u,v; scanf("%d%d",&u,&v);
g[u].push_back(v); g[v].push_back(u);
}
for(int i=;i<=n;i++) scanf("%1d",&a[i]);
} void work(){
pw2[] = ;
for(int i=;i<=m*;i++) pw2[i] = pw2[i-]*%mod;
for(int i=;i<=n;i++) {
if(!dfn[i]) {
dfs(i,); cnt++;
if(sz[i]) flag++;
}
}
if(flag == ){ans[] = pw2[m-n+cnt];}
if(flag >= ){
for(int i=;i<=n;i++) printf("0 ");
puts("");
return;
}
for(int i=;i<=n;i++) {
if(pa[i] == ){dfs2(i,,sz[i]);}
}
for(int i=;i<=n;i++) printf("%d ",ans[i]);
puts("");
} int main(){
int T; scanf("%d",&T);
while(T--){
init();
read();
work();
}
return ;
}

Luogu4494 [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. [HAOI2018]反色游戏

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

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

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

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

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

随机推荐

  1. EF获取多个数据集以及MySQL分页数据查询优化

    背景:MySQL分页查询语句为 ,10; 一般页面还会获取总条数,这时候还需要一条查询总条数语句 , 这样数据库需要执行两次查询操作.MySQL提供了SQL_CALC_FOUND_ROWS追踪总条数的 ...

  2. WPF 自定义 ImageButton

    控件源码: public class ImageButton : Button    {        public ImageButton() {        } public string No ...

  3. C# T4 模板 数据库实体类生成模板(带注释,娱乐用)

     说明:..,有些工具生成实体类没注释,不能和SqlServer的MS_Description属性一起使用,然后照着网上的资源,随便写了个生成模板,自娱自乐向,其实卵用都没有参考教程    1.htt ...

  4. C#的dapper使用

    Dapper是一款轻量级ORM工具(Github).如果你在小的项目中,使用Entity Framework.NHibernate 来处理大数据访问及关系映射,未免有点杀鸡用牛刀.你又觉得ORM省时省 ...

  5. Spring Boot Security OAuth2 实现支持JWT令牌的授权服务器

    概要 之前的两篇文章,讲述了Spring Security 结合 OAuth2 .JWT 的使用,这一节要求对 OAuth2.JWT 有了解,若不清楚,先移步到下面两篇提前了解下. Spring Bo ...

  6. 戏说程序猿之cannot find the object

    “别开玩笑了,程序员哪里需要对象!” 程序员难找对象原因无非如下: 1.工作时间长,恋爱时间少 2.性格偏于内向,不主动 3.不注意个人形象 程序员爱情观: 爱情就是死循环,一旦执行就陷进去了: 爱上 ...

  7. 利用Redis keyspace notification(键空间通知)实现过期提醒

    一.序言: 本文所说的定时任务或者说计划任务并不是很多人想象中的那样,比如说每天凌晨三点自动运行起来跑一个脚本.这种都已经烂大街了,随便一个 Crontab 就能搞定了. 这里所说的定时任务可以说是计 ...

  8. java集合(1)

    java集合类存放于java.util包里,只能存放对象,存放的是对象的引用,可以是不同类型,不限数量的数据类型. 顶层接口:Iterator(迭代器),Map Iterator:核心方法hasNex ...

  9. python的学习笔记01_2变量 常量 注释 用户交互 格式化输出

    变量是什么? 变量的作用  Variables are used to store information to be referenced and manipulated in a computer ...

  10. iBatis第一章:基础知识概述 & MVC思想

    一.java是一门十分受开发人员欢迎的语言,在开发语言排行榜中名列前茅,人们对其看法不尽相同,就我自身感受而言,我觉得java语言的主要优势体现在如下几方面:1.java属于开源语言,开发人员可以找到 ...