[BZOJ5303] [HAOI2018] 反色游戏
题目链接
LOJ:https://loj.ac/problem/2524
BZOJ:https://lydsy.com/JudgeOnline/problem.php?id=5303
洛谷:https://www.luogu.org/problemnew/show/P4494
Solution
精神污染
假设所有点都是连通的,很显然如果黑点个数为奇数个则无解,否则可以证明一定有解。
那么随便整出一棵生成树,然后反色一些边使其合法,显然一棵树只有一种情况。
考虑非树边的贡献,如果非树边不反色显然无影响,否则可以把这条边两个端点连成的链上所有边都反色,那么可以发现当前方案还是合法的。
那么每条非树边都可选可不选,总方案数就是\(2^{m-n+1}\)。
注意到原图不一定联通,考虑每个联通块,方案数相乘,可以得到总方案数为\(2^{m-n+p}\),其中\(p\)为联通块个数。
注意每个联通块都必须合法,否则无解。
对于删点,我们分情况讨论:
- 如果当前点是孤立的,对答案无影响。
- 当前点不是割点,那么对答案影响无非就是点数少\(1\),边数少了点,直接算就好了。
- 对于割点,我们需要判割完之后剩下的联通块合不合法,答案也差不多。
这题最难写的地方就是判无解....不过据说建圆方树好写点?
细节很多...
#include<bits/stdc++.h>
using namespace std;
void read(int &x) {
x=0;int f=1;char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if(ch=='-') f=-f;
for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';x*=f;
}
void print(int x) {
if(x<0) putchar('-'),x=-x;
if(!x) return ;print(x/10),putchar(x%10+48);
}
void write(int x) {if(!x) putchar('0');else print(x);putchar(' ');}
#define lf double
#define ll long long
const int maxn = 2e5+10;
const int inf = 1e9;
const lf eps = 1e-8;
const int mod = 1e9+7;
char s[maxn];
int n,m,head[maxn],tot,dfn[maxn],dfn_cnt,low[maxn],bo[maxn],cnt[maxn],rt,c[maxn],is[maxn],bel[maxn],deg[maxn],sz[maxn],pw[maxn],siz[maxn];
struct edge{int to,nxt;}e[maxn<<1];
void add(int u,int v) {e[++tot]=(edge){v,head[u]},head[u]=tot,deg[v]++;}
void ins(int u,int v) {add(u,v),add(v,u);}
void tarjan(int x,int fa) {
dfn[x]=low[x]=++dfn_cnt,bel[x]=rt,bo[x]=1,siz[x]=sz[x];int flag=0;
for(int v,i=head[x];i;i=e[i].nxt) {
if((v=e[i].to)==fa) continue;flag++;
if(!dfn[v]) tarjan(v,x),low[x]=min(low[x],low[v]),sz[x]+=sz[v];
else {low[x]=min(low[x],dfn[v]);continue;}
if(low[v]>=dfn[x]) bo[x]&=!(sz[v]&1),cnt[x]++,is[x]=1,siz[x]+=sz[v];
}
if(fa) cnt[x]++;if(!flag) is[x]=1;
}
void solve() {
read(n),read(m);
for(int i=1,x,y;i<=m;i++) read(x),read(y),ins(x,y);
scanf("%s",s+1);for(int i=1;i<=n;i++) sz[i]=c[i]=s[i]-'0';
pw[0]=1;for(int i=1;i<=m;i++) pw[i]=2ll*pw[i-1]%mod;
int p=0,odd=0;
for(int i=1;i<=n;i++)
if(!dfn[i]) rt=i,tarjan(i,0),p++,odd+=sz[i]&1;
for(int i=1;i<=n;i++) bo[i]&=!((sz[bel[i]]-siz[i])&1);
write(odd?0:pw[m-n+p]);
for(int i=1;i<=n;i++) {
if(!deg[i]) write(odd-sz[i]?0:pw[m-n+p]);
else if(!is[i]) write(odd-(sz[bel[i]]&1)+((sz[bel[i]]-c[i])&1)?0:pw[m-deg[i]-n+p+1]);
else write(bo[i]&&!(odd-(sz[bel[i]]&1))?pw[m-deg[i]-n+cnt[i]+p]:0);
}puts("");
}
#define clr(x) memset(x,0,4*(n+5))
void clear() {
clr(dfn),clr(low),clr(bel),clr(deg);
clr(sz),clr(bo),clr(is),clr(head),clr(cnt),clr(siz);
tot=0;
}
int main() {
int t;read(t);while(t--) solve(),clear();
return 0;
}
[BZOJ5303] [HAOI2018] 反色游戏的更多相关文章
- [BZOJ5303][HAOI2018]反色游戏(Tarjan)
暴力做法是列异或方程组后高斯消元,答案为2^自由元个数,可以得60分.但这个算法已经到此为止了. 从图论的角度考虑这个问题,当原图是一棵树时,可以从叶子开始唯一确定每条边的选择情况,所以答案为1. 于 ...
- 【BZOJ5303】[HAOI2018]反色游戏(Tarjan,线性基)
[BZOJ5303][HAOI2018]反色游戏(Tarjan,线性基) 题面 BZOJ 洛谷 题解 把所有点全部看成一个\(01\)串,那么每次选择一条边意味着在这个\(01\)串的基础上异或上一个 ...
- bzoj 5393 [HAOI2018] 反色游戏
bzoj 5393 [HAOI2018] 反色游戏 Link Solution 最简单的性质:如果一个连通块黑点个数是奇数个,那么就是零(每次只能改变 \(0/2\) 个黑点) 所以我们只考虑偶数个黑 ...
- P4494 [HAOI2018]反色游戏
P4494 [HAOI2018]反色游戏 题意 给你一个无向图,图上每个点是黑色或者白色.你可以将一条边的两个端点颜色取反.问你有多少种方法每个边至多取反一次使得图上全变成白色的点. 思路 若任意一个 ...
- 【loj#2524】【bzoj5303】 [Haoi2018]反色游戏(圆方树)
题目传送门:loj bzoj 题意中的游戏方案可以转化为一个异或方程组的解,将边作为变量,点作为方程,因此若方程有解,方程的解的方案数就是2的自由元个数次方.我们观察一下方程,就可以发现自由元数量=边 ...
- [HAOI2018]反色游戏
[Luogu4494] [BZOJ5303] [LOJ2524] LOJ有数据就是好 原题解,主要是代码参考 对于每一个联通块(n个点),其他的边一开始随便选,只需要n-1条边就可以确定最终结果. 所 ...
- bzoj 5303: [Haoi2018]反色游戏
Description Solution 对于一个有偶数个黑点的连通块,只需要任意两两配对,并把配对点上的任一条路径取反,就可以变成全白了 如果存在奇数个黑点的连通块显然无解,判掉就可以了 如果有解, ...
- Luogu4494 [HAOI2018]反色游戏 【割顶】
首先发现对于一个联通块有奇数个黑点,那么总体来说答案无解.这个很容易想,因为对每个边进行操作会同时改变两个点的颜色,异或值不变. 然后一个朴素的想法是写出异或方程进行高斯消元. 可以发现高斯消元的过程 ...
- 洛谷P4494 [HAOI2018]反色游戏(tarjan)
题面 传送门 题解 我们先来考虑一个联通块,这些关系显然可以写成一个异或方程组的形式,形如\(\oplus_{e\in edge_u}x_e=col_u\) 如果这个联通块的黑色点个数为奇数,那么显然 ...
随机推荐
- ATextAppearance.AppCompat.Small not found
今天编译的代码的时候,刚才还是好的,后来吃个饭回来,就不行了. 报错如下: AGPBI: {"kind":"error","text":&q ...
- ORB-SLAM(六)MapPoint与Map
地图点可以通过关键帧来构造,也可以通过普通帧构造,但是最终,必须是和关键帧对应的,通过普通帧构造的地图点只是临时被Tracking用来追踪用的. 构造函数(地图点3D坐标及其参考帧): // 参考帧是 ...
- WeTest功能优化第2期:云真机智能投屏,调试告别鼠标
第2期功能优化目录 [云真机视频映射]云真机画面本地映射[兼容性测试报告]新增问题机型聚类功能[新增Android9.0]同步上线最新安卓系统 本期介绍的云测产品功能优化,既有重磅级技术突破,也有报告 ...
- VIN码识别:让VIN码采集so easy!
近几年汽车后市场呈喷井式发展,在过去的半年,汽车后市场规模已高达万亿级,产业前景广阔,与此同时行业运营也受信息区域化.数据不统一的制约,让企业面临着效率低下.规模化运行困难的痛点. 在汽车配件市场中, ...
- JAVA基础:ArrayList和LinkedList区别
1.ArrayList是实现了基于动态数组的数据结构,LinkedList基于链表的数据结构. 2.对于随机访问get和set,ArrayList觉得优于LinkedList,因为LinkedList ...
- Java进阶知识点:不要只会写synchronized - JDK十大并发编程组件总结
一.背景 提到Java中的并发编程,首先想到的便是使用synchronized代码块,保证代码块在并发环境下有序执行,从而避免冲突.如果涉及多线程间通信,可以再在synchronized代码块中使用w ...
- 最小生成树与Prim算法
最小生成树(MST) 定义 首先是一棵树(废话 其次没有回路(废话 包含全部顶点和V-1条边 边的权重和最小!!!!! 所以如果是单棵最小生成树,至少说明图是连通的.不然就是森林. 生成思路 既然是根 ...
- Tensorflow - Implement for generating some 3-dimensional phony data and fitting them with a plane.
Coding according to TensorFlow 官方文档中文版 import tensorflow as tf import numpy as np ''' Intro. for thi ...
- UVa 1583 - Digit Generator 解题报告 - C语言
1.题目大意 如果a加上a的各个数字之和得到b,则说a是b的生成元.给出n其中$1\le n\le 100000$,求其最小生成元,若没有解则输出0. 2.思路 使用打表的方法打出各个数字a对应的b, ...
- Python3 小工具-僵尸扫描
僵尸机的条件: 1.足够闲置,不与其他机器进行通讯 2.IPID必须是递增的,不然无法判断端口是否开放 本实验僵尸机选择的是Windows2003 from scapy.all import * im ...