P4494-[HAOI2018]反色游戏【圆方树】
正题
题目链接:https://www.luogu.com.cn/problem/P4494
题目大意
给出\(n\)个点\(m\)条边的一张无向图,节点有\(0/1\),每条边可以选择是否取反两边的点。
开始求将所有节点变为\(0\)的方案,然后对于每个点询问删去这个点之后的方案
\(1\leq T\leq 5,1\leq n,m\leq 10^5\)
解题思路
图的比较麻烦,先考虑树上的,那么每条边取不取反取决于它连接的子节点的黑白,但是根节点却无法这么调整。所以如果黑色个数为奇数个那么方案为\(0\),否则方案为\(1\)。
然后考虑一张连通图,考虑对于图中的一个生成树来说,无论非生成树上的边是否取反,都可以用这棵生成树调整回来,也就是如果黑色为奇数个方案为\(0\),否则方案为\(2^{m-n+1}\)。
因为原图不一定连通,设连通块个数为\(k\),那么第一问答案就是\(2^{m-n+k}\)(每个连通块的黑色个数为奇数个)。
然后第二问,其实就是去掉这条边之后会分割一个连通块以影响答案。
建立广义圆方树,统计每个点删去后会多产生的连通块数量以及是否有分割出来的连通块的黑色个数为奇数。
顺带一提的是需要特判如果有两个或者以上的连通块黑色为奇数个,那么全都无解,否则只有可能删除掉黑色奇数连通块里的点。
时间复杂度\(O(Tn)\)
code
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<stack>
#include<vector>
#define ll long long
using namespace std;
const ll N=2e5+10,P=1e9+7;
ll Z,n,m,dfc,sum,cnt,st[N],deg[N];
ll dfn[N],low[N],pw[N],siz[N];
bool tag[N],nok[N],v[N];
stack<ll> s;char t[N];
vector<ll>G[N],T[N];
void tarjan(ll x){
dfn[x]=low[x]=++dfc;sum+=(t[x]=='1');
s.push(x);st[++st[0]]=x;
for(ll i=0;i<G[x].size();i++){
ll y=G[x][i];
if(!dfn[y]){
tarjan(y);
low[x]=min(low[x],low[y]);
if(low[y]==dfn[x]){
ll k;++cnt;
do{
k=s.top();s.pop();deg[k]--;
T[cnt].push_back(k);
T[k].push_back(cnt);
}while(k!=y);
T[cnt].push_back(x);
T[x].push_back(cnt);
deg[x]--;
}
}
else low[x]=min(low[x],dfn[y]);
}
return;
}
void dfs(ll x){
v[x]=1;st[++st[0]]=x;
siz[x]=(x<=n)&(t[x]=='1');
for(ll i=0;i<T[x].size();i++){
ll y=T[x][i];
if(v[y])continue;
dfs(y);siz[x]+=siz[y];
if(siz[y]&1)nok[x]=1;
}
return;
}
signed main()
{
scanf("%lld",&Z);pw[0]=1;
for(ll i=1;i<N;i++)pw[i]=pw[i-1]*2%P;
while(Z--){
dfc=0;
memset(deg,0,sizeof(deg));
memset(nok,0,sizeof(nok));
memset(tag,0,sizeof(tag));
memset(dfn,0,sizeof(dfn));
memset(v,0,sizeof(v));
while(!s.empty())s.pop();
scanf("%lld%lld",&n,&m);
for(ll i=1;i<=2*n;i++)
T[i].clear(),G[i].clear();
for(ll i=1;i<=m;i++){
ll x,y;scanf("%lld%lld",&x,&y);
G[x].push_back(y);deg[x]++;
G[y].push_back(x);deg[y]++;
}
scanf("%s",t+1);cnt=n;
ll one=0,k=0;
for(ll i=1;i<=n;i++){
if(dfn[i])continue;
st[0]=sum=0;
tarjan(i);k++;
if(sum&1){
for(ll j=1;j<=st[0];j++)
tag[st[j]]=1;
one++;
}
}
if(one>1){
for(ll i=0;i<=n;i++)printf("0 ");
putchar('\n');continue;
}
else if(one)printf("0 ");
else printf("%lld ",pw[m-n+k]);
for(ll i=1;i<=n;i++){
if(v[i])continue;
st[0]=0;dfs(i);
for(ll j=1;j<=st[0];j++)
if((siz[i]-siz[st[j]])&1)nok[st[j]]=1;
}
for(ll i=1;i<=n;i++)
if(nok[i]||(one&&!tag[i]))printf("0 ");
else printf("%lld ",pw[m-n+k-deg[i]]);
putchar('\n');
}
return 0;
}
P4494-[HAOI2018]反色游戏【圆方树】的更多相关文章
- P4494 [HAOI2018]反色游戏
P4494 [HAOI2018]反色游戏 题意 给你一个无向图,图上每个点是黑色或者白色.你可以将一条边的两个端点颜色取反.问你有多少种方法每个边至多取反一次使得图上全变成白色的点. 思路 若任意一个 ...
- 洛谷P4494 [HAOI2018]反色游戏(tarjan)
题面 传送门 题解 我们先来考虑一个联通块,这些关系显然可以写成一个异或方程组的形式,形如\(\oplus_{e\in edge_u}x_e=col_u\) 如果这个联通块的黑色点个数为奇数,那么显然 ...
- bzoj 5393 [HAOI2018] 反色游戏
bzoj 5393 [HAOI2018] 反色游戏 Link Solution 最简单的性质:如果一个连通块黑点个数是奇数个,那么就是零(每次只能改变 \(0/2\) 个黑点) 所以我们只考虑偶数个黑 ...
- 【BZOJ5303】[HAOI2018]反色游戏(Tarjan,线性基)
[BZOJ5303][HAOI2018]反色游戏(Tarjan,线性基) 题面 BZOJ 洛谷 题解 把所有点全部看成一个\(01\)串,那么每次选择一条边意味着在这个\(01\)串的基础上异或上一个 ...
- bzoj5315/luoguP4517 [SDOI2018]战略游戏(圆方树,虚树)
bzoj5315/luoguP4517 [SDOI2018]战略游戏(圆方树,虚树) bzoj Luogu 题目描述略(太长了) 题解时间 切掉一个点,连通性变化. 上圆方树. $ \sum |S| ...
- 【loj#2524】【bzoj5303】 [Haoi2018]反色游戏(圆方树)
题目传送门:loj bzoj 题意中的游戏方案可以转化为一个异或方程组的解,将边作为变量,点作为方程,因此若方程有解,方程的解的方案数就是2的自由元个数次方.我们观察一下方程,就可以发现自由元数量=边 ...
- [BZOJ5303] [HAOI2018] 反色游戏
题目链接 LOJ:https://loj.ac/problem/2524 BZOJ:https://lydsy.com/JudgeOnline/problem.php?id=5303 洛谷:https ...
- BZOJ5329:[SDOI2018]战略游戏(圆方树,虚树)
Description 省选临近,放飞自我的小Q无心刷题,于是怂恿小C和他一起颓废,玩起了一款战略游戏. 这款战略游戏的地图由n个城市以及m条连接这些城市的双向道路构成,并且从任意一个城市出发总能沿着 ...
- [SDOI2018]战略游戏 圆方树,树链剖分
[SDOI2018]战略游戏 这题是道路相遇(题解)的升级版,询问的两个点变成了\(S\)个点. LG传送门 还是先建出圆方树,考虑对于询问的\(S\)个点,答案就是圆方树上能包含这些点的最小连通块中 ...
- BZOJ.5329.[SDOI2018]战略游戏(圆方树 虚树)
题目链接 显然先建圆方树,方点权值为0圆点权值为1,两点间的答案就是路径权值和减去起点终点. 对于询问,显然可以建虚树.但是只需要计算两关键点间路径权值,所以不需要建出虚树.统计DFS序相邻的两关键点 ...
随机推荐
- Python爬虫(二)——发送请求
1. requests库介绍 在python中有许多支持发送的库.比如:urlib.requests.selenium.aiohttp--等.但我们当前最常用的还是requests库,这个库是基于 ...
- .Net 5 新特性之--支持字典在迭代中进行修改
我们都知道以前字典迭代中是不支持动态修改的[否则会报错::"Collection was modified; enumeration operation may not execute.&q ...
- MAC下Jetbrains编译器无法打开问题解决
这段时间不知道怎么回事,每次打开Rider必定闪退,毫无头绪,只好暂时放弃使用Rider,试用了一段时间Visual Studio. 可惜...虽然大学时候觉得VS天下第一,但是用惯了JB的编译器,再 ...
- TypeScript 中枚举类型的理解?应用场景?
一.是什么 枚举是一个被命名的整型常数的集合,用于声明一组命名的常数,当一个变量有几种可能的取值时,可以将它定义为枚举类型 通俗来说,枚举就是一个对象的所有可能取值的集合 在日常生活中也很常见,例如表 ...
- HCNP Routing&Switching之OSPF LSA类型(二)
前文我们了解了OSPF的一类.二类.三类LSA,回顾请参考https://www.cnblogs.com/qiuhom-1874/p/15209829.html:今天我们来聊一聊OSPF的四类和五类L ...
- nginx 开启,关闭,重启
2021-08-191. 启动 # 判断配置文件是否正确 cd /usr/local/nginx/sbin ./nginx -t # 启动 cd usr/local/nginx/sbin ./ngin ...
- Linux CentOS7 安装配置 IPtables
2021-08-11 1. 前言 防火墙其实就是实现 Linux 下访问控制功能的,分为硬件和软件的防火墙两种类型.无论在何网络中,防火墙工作的地方一定是网络的边缘.防火墙的策略.规则就是去定义防火墙 ...
- Spring Boot 入门系列(二十四)多环境配置,3分钟搞定!
之前讲过Spring Boot 的系统配置和自定义配置,实现了按照实际项目的要求配置系统的相关熟悉.但是,在实际项目开发过程中,需要面对不同的环境,例如:开发环境,测试环境,生产环境.各个环境的数据库 ...
- 从零开始实现简单 RPC 框架 9:网络通信之心跳与重连机制
一.心跳 什么是心跳 在 TPC 中,客户端和服务端建立连接之后,需要定期发送数据包,来通知对方自己还在线,以确保 TPC 连接的有效性.如果一个连接长时间没有心跳,需要及时断开,否则服务端会维护很多 ...
- SQL-UPDATE触发器练习
&练习一 如下所示三张表( student,grade,student_updata_before ): student表 grade表 Student_update_before表 # 触发 ...