T15 Apio2011 方格染色
20分算法:暴力枚举每个点
抄题解做的还是好好写总结吧
把红色视为0,蓝色视为1
假设有一个表格如下:
A | C | E | G |
---|---|---|---|
B | D | F | H |
根据题设有(Aigoplus Bigoplus Cigoplus D=Cigoplus Digoplus Eigoplus F=1)
然后有(Aigoplus Bigoplus Eigoplus F=0)
而(Eigoplus Figoplus Gigoplus H=1)
所以(Aigoplus Bigoplus Gigoplus H=1)
再放到一般性的表格中
A | ...... | C |
---|---|---|
B | ...... | D |
I | ...... | J |
....... | ...... | ....... |
E | ...... | G |
F | ...... | H |
当C,D处于奇数列的时候,有:
(Aigoplus Bigoplus Cigoplus D=0,Eigoplus Figoplus Gigoplus H=0,Bigoplus Iigoplus Digoplus J=0)
所以有:
(Aigoplus Cigoplus Iigoplus J=0)
推出
(Aigoplus Cigoplus Figoplus H=0)
即:
(Aigoplus H=Cigoplus F)
当C,D在偶数列上,有:
(Aigoplus Bigoplus Cigoplus D=1,Eigoplus Figoplus Gigoplus H=1,Bigoplus Iigoplus Digoplus J=1)
当H在奇数行:
(1igoplus Aigoplus H=Cigoplus F)
当H在偶数行:
(Aigoplus H=Cigoplus F)
设H坐标为((i,j)),A坐标为((1,1))
则有:
(if(i mod 2==0 and j mod 2==0) 1igoplus(1,1)igoplus(i,j) ==(1,j)igoplus(i,1))
(else (1,1)igoplus(i,j) ==(1,j)igoplus(i,1))
而显然如果确定了第一排,第一列,那么可以确定整张图。
所以可以枚举((1,1))的值,然后合并所有相关的集合,去掉已知点的集合,剩下的集合个数为(cnt),则答案为(2^{cnt})。
把(1,1)为0,1的答案加起来即可
合并集合的时候使用扩展域并查集维护到根节点的异或值。
#include <cstdio>
#include <cstring>
#include <cstdlib>
#define R register
#define ll long long
const int MAXN=1e6+10;
const int Mod=1e9;
inline int read()
{
int x=0,f=1;
char a=getchar();
for(;a>'9'||a<'0';a=getchar()) if(a=='-') f=-1;
for(;a>='0'&&a<='9';a=getchar()) x=x*10+a-'0';
return x*f;
}
int n,m,k,flg=-1,cnt,Size;
int fa[MAXN];
int X[MAXN],Y[MAXN],Z[MAXN];
inline int find(int x) { return x==fa[x]?x:fa[x]=find(fa[x]); }
inline void merge(int x,int y)
{
int fx=find(x),fy=find(y);
fa[fx]=fy;cnt-=fx!=fy;
}
inline void Init()
{
n=read();m=read();k=read();
Size=n+m-1;
for(R int i=1;i<=k;i++)
{
X[i]=read();Y[i]=read();Z[i]=read();
if(X[i]==1&&Y[i]==1) flg=Z[i];
}
}
inline int id(int x) { return x==1?1:m+x-1; }
int vis[MAXN];
int ans=0;
inline void Solve(bool delta)
{
for(R int i=1;i<=Size*2;i++) fa[i]=i;
cnt=Size*2;
for(R int i=1;i<=k;i++)
{
int a=X[i],b=Y[i];
if(a==1&&b==1) continue;
if(a%2==0&&b%2==0)
{
if(1^delta^Z[i])
{
merge(id(a),b+Size);merge(id(a)+Size,b);
}
else
{
merge(id(a),b);merge(id(a)+Size,b+Size);
}
}
else
{
if(delta^Z[i])
{
merge(id(a),b+Size);
merge(id(a)+Size,b);
}
else
{
merge(id(a),b);
merge(id(a)+Size,b+Size);
}
}
if(find(b)==find(b+Size)||find(id(a))==find(id(a)+Size))
{
printf("0
");
exit(0);
}
}
memset(vis,0,sizeof(vis));
cnt--;
vis[find(1)]=1;
for(R int i=1;i<=k;i++)
{
if(X[i]==1&&vis[find(Y[i])]==0)
{
vis[find(Y[i])]=1;
cnt--;
}
else
if(Y[i]==1&&vis[find(id(X[i]))]==0)
{
vis[find(id(X[i]))]=1;
cnt--;
}
}
cnt>>=1;
int tmp=1;
for(R int i=1;i<=cnt;i++) tmp=(tmp<<1)>=Mod?(tmp<<1)-Mod:(tmp<<1);
ans=ans+tmp>=Mod?ans+tmp-Mod:ans+tmp;
}
int main()
{
Init();
if(flg==-1) Solve(0),Solve(1);
if(flg==0) Solve(0);
if(flg==1) Solve(1);
printf("%d
",ans);
return 0;
}