[HAOI2018] 反色游戏

 

    做法超好想,细节调一辈子。 估计这句话最适合这个题了hhhh。

    首先一个很显然的想法:把边设成未知数,对点列异或方程,最后的解的个数就是 2^自由元 。 

    不过如果某个联通分量里有奇数个黑点,那么问题无解。然后我来证明一下:

        把一个联通分量里所有点代表的方程都异或起来,因为这个联通分量里的边恰好在其两个端点的行是1,所以最后异或出来的方程前m列都是0,如果有奇数个黑点的话,最后一列就是1,那么无解(不能0=1吧hhh)。

     直接高斯消元是 O(N^3) 的,肯定要gg了啊 (不过竟然给了60分消元hhh,出题人太良心了)。

 

    不过一般这种题我们如果按照特定的顺序消(shou)元(wan)的话,会得到一些很良心的结论(具体参考 bzoj 文艺计算姬)。

    考虑 边1 的两个端点 u,v,因为最后只能有一行对应在这列是1,所以我们就把第v行变成 第u行 异或 第v行,可以发现得到的新行的意义是:u点和v点合并后的新点(因为边1被消去了,正好对应一个点没有自环)。

    当然,如果我们考虑边i的时候,发现它的两个端点已经在一个联通分量里了,那么这条边就是自由元,因为我们现在在高斯消元的矩阵里已经找不到第i列是1的行了。

    我们把上述做法扩展一下,就可以得到一个能够算出全局答案的做法:初始答案是1,并查集维护连通性,如果尝试合并失败,那么答案*2。

 

    但是这还不够棒,因为本题要求出 删去 每一个点之后的答案。

    不过上述做法还是有点多余,因为我们仔细想一下它的过程,就可以发现其实答案就是  2^(m-n + 联通分量个数)。

 

    所以,现在我们还剩下的最大的问题就是:如何维护,删去一个点之后,可能带来的某个联通分量的黑点变成奇数个(对应无解)   和  联通分量个数的变动 (对应图的连通性)  的影响???

     

    跑个tarjan求割点的算法就好啦,第一个问题有些复杂,我们先解决一下第二个问题。

     

    第二个问题无非可以分以下几种情况:

        1.删去的节点是一个孤立顶点: 这种情况下 会使图中联通分量个数-1.

        2.删去一个割点: 这种情况可以在tarjan的时候顺带统计,不过如果这个点是dfs树的根的话答案还要再-1(因为上面就没有新的联通分量了)

        3.删去一个非孤立顶点且非割点:没有影响hhhh。

    但是不要忘了,删一个点的时候不仅会对联通分量个数产生影响,还会对图的总边数-点数 产生影响。

 

    至于第一个问题,我们可以在tarjan的时候处理出:删去一个点之后,图中 有奇数个黑点的联通分量 的个数的 变化量  derta[i]。

    怎么处理呢?   

    首先如果这个点是割点的话:

        1.如果这个点在的联通分量本来就有 奇数个黑点 的话,那么先让 derta[i]-- ,因为删去这个点会破坏这个联通分量。

        2.再让derta[i]加上 删去这个点之后, 在dfs树上不能回到祖先的有奇数个黑点的联通分量个数 。

        3.如果 dfs树根所在的联通分量在删去这个点之后也有奇数个黑点,那么 derta[i]++。

    如果这个点i是非割点且非孤立顶点的话,那么derta[i] = color[i] == 1 ? (color[this tree] == 1 ? -1 : 1) : 0 .

    如果这个点是孤立顶点的话,那么derta[i] = color[i] == 1 ?  -1 : 0.

 

    (呼。。。。。终于写完了累死我了。。。。)

 

#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int maxn=200005,ha=1000000007;
inline int add(int x,int y){ x+=y; return x>=ha?x-ha:x;}
int to[maxn*2],ne[maxn*2],hd[maxn],num,n,m,T,ans,Sub[maxn];
int Xor[maxn],NewC[maxn],NewB[maxn],BC,ci[maxn];
int dfn[maxn],low[maxn],dc,tp,st[maxn],D[maxn];
bool iscut[maxn];
char S[maxn];
inline void addline(int x,int y){ to[++num]=y,ne[num]=hd[x],hd[x]=num,D[x]++;}
inline void init(){
	memset(hd,0,sizeof(hd)),num=0;
	memset(NewB,0,sizeof(NewB));
	memset(NewC,0,sizeof(NewC));
	memset(D,0,sizeof(D));
	memset(iscut,0,sizeof(iscut));
	memset(dfn,0,sizeof(dfn));
	memset(Sub,0,sizeof(Sub));
	dc=BC=0;
}
inline int read(){
	int x=0; char ch=getchar();
	for(;!isdigit(ch);ch=getchar());
	for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';
	return x;
}
void W(int x){ if(x>=10) W(x/10); putchar(x%10+'0');}

void dfs(int x,int fa){
	st[++tp]=x,dfn[x]=low[x]=++dc;
	int cd=0; Sub[x]=Xor[x];
	for(int i=hd[x];i;i=ne[i]) if(to[i]!=fa)
	    if(!dfn[to[i]]){
	    	cd++,dfs(to[i],x);
			Xor[x]^=Xor[to[i]];
			low[x]=min(low[x],low[to[i]]);
			
	    	if(low[to[i]]>=dfn[x]){
	    		Sub[x]^=Xor[to[i]];
			    iscut[x]=1,NewC[x]++;
			    if(Xor[to[i]]) NewB[x]++;
			}
		}
		else low[x]=min(low[x],dfn[to[i]]);
	
	if(cd==1&&fa<0) iscut[x]=NewB[x]=NewC[x]=0;
	else if(fa<0) NewC[x]--;
}

inline void solve(){
	for(int i=1;i<=n;i++) if(S[i]=='1') Xor[i]=1; else Xor[i]=0;
	ans=m-n;
	for(int i=1;i<=n;i++) if(!dfn[i]){
		ans++;
		dfs(i,-1);
		if(Xor[i]) BC++;
		if(!hd[i]) NewC[i]=-1;
		
		for(;tp;tp--) if(iscut[st[tp]]){
			if(Xor[i]^Sub[st[tp]]) NewB[st[tp]]++;
			if(Xor[i]==1) NewB[st[tp]]--;
		}
		else NewB[st[tp]]=S[st[tp]]=='1'?(Xor[i]?-1:1):0;
	}
	
	if(BC) W(0); else W(ci[ans]); putchar(' ');
	for(int i=1;i<=n;i++){
		BC+=NewB[i],ans+=NewC[i]-D[i]+1;
		if(!BC) W(ci[ans]); else W(0); putchar(' ');
		BC-=NewB[i],ans-=NewC[i]-D[i]+1;
	}	
	puts("");
}

int main(){
	ci[0]=1; for(int i=1;i<=200000;i++) ci[i]=add(ci[i-1],ci[i-1]);
	T=read();
	while(T--){
		int uu,vv;
		init(),n=read(),m=read();
		for(int i=1;i<=m;i++) uu=read(),vv=read(),addline(uu,vv),addline(vv,uu);
		scanf("%s",S+1),solve();
	}
	return 0;
}

  

posted @ 2018-04-29 11:30  蒟蒻JHY  阅读(1040)  评论(1编辑  收藏  举报