欢迎访问~原文出处——博客园-zhouzhendong

去博客园看该题解


题目传送门 - BZOJ2303


题意概括

  现在有一个N*M矩阵,矩阵上只能填数字0或1 
现在矩阵里已经有一些格子被填写了数字,询问是否存在一种填写方案使得「任意一个2*2的矩阵异或和为1」,输出方案总数


题解

  我们发现当我们已经确定(1,1)的颜色为1的时候:

  我们知道c(i,j)。

  那么如果i和j都是偶数,那么就有c(1,1)^c(i,1)^c(1,j)^c(i,j)==1

  否则就是0。

  因为假设s(i,j)表示以i,j为左上角的2*2矩阵异或起来,那么:

  S(1,1)^S(1,2)^...^S(1,j)^S(2,1)^S(2,2)^...^S(i-1,j-1)=c(1,1)^c(i,1)^c(1,j)^c(i,j)。

  而左式就等于(i-1)*(j-1)个1异或。

  这是一个好东西。

  然后我们枚举(1,1)的颜色,然后对于每一个C(i,j)就可以得到一组C(i,1)和C(1,j)的关系。

  但是当i=1或者j=1的时候,是得到一个C(i,1)或者C(1,j)的答案。

  最后得到关系之后,只需要看看是否矛盾。

  然后统计连通块数,就是自由元的总数,每一个自由元有2种取值,于是答案显而易见。

  注意输入有i=j=1的情况要特判。

  细节有点多。


代码

#include <cstring>
#include <algorithm>
#include <cstdio>
#include <cmath>
#include <cstdlib>
using namespace std;
const int N=1000005,mod=1e9;
int n,m,ad,k,v[N*4],fa[N*4];
struct color{
int a,b,c;
}co[N];
int Turn(int a,int b){
return a==1?(b-1):(a-1+m-1);
}
int getf(int k){
return fa[k]==k?k:fa[k]=getf(fa[k]);
}
int solve(){
memset(v,-1,sizeof v);
for (int i=1;i<=ad*2;i++)
fa[i]=i;
for (int i=1;i<=k;i++){
int a=co[i].a,b=co[i].b,c=co[i].c;
if (a==1&&b==1)
continue;
if (a==1||b==1){
v[Turn(a,b)]=c,v[Turn(a,b)+ad]=c^1;
continue;
}
int A=Turn(a,1),B=Turn(1,b);
int res=(!(a&1)&&!(b&1))^c;
if (res){
if (getf(A)==getf(B))
return 0;
fa[getf(A)]=getf(B+ad);
fa[getf(B)]=getf(A+ad);
}
else {
if (getf(A)==getf(B+ad))
return 0;
fa[getf(A)]=getf(B);
fa[getf(A+ad)]=getf(B+ad);
}
}
for (int i=1;i<=ad*2;i++){
if (v[i]==-1)
continue;
if (v[getf(i)]==-1)
v[getf(i)]=v[i];
else if (v[getf(i)]!=v[i])
return 0;
}
int res=1,ans=0;
for (int i=1;i<=ad*2;i++)
if (getf(i)==i&&v[i]==-1)
ans++;
for (ans>>=1;ans--;)
res=res*2%mod;
return res;
}
int main(){
scanf("%d%d%d",&n,&m,&k);
ad=n+m-2;
int flag=-1;
for (int i=1;i<=k;i++){
scanf("%d%d%d",&co[i].a,&co[i].b,&co[i].c);
if (co[i].a==1&&co[i].b==1)
flag=co[i].c;
}
int ans1=solve();
for (int i=1;i<=k;i++)
if (co[i].a>1&&co[i].b>1)
co[i].c^=1;
int ans2=solve();
int ans=0;
if (flag==-1)
ans=(ans1+ans2)%mod;
else
ans=flag?ans2:ans1;
printf("%d",ans);
return 0;
}

  

BZOJ2303 [Apio2011]方格染色 并查集的更多相关文章

  1. BZOJ 2303: [Apio2011]方格染色 [并查集 数学!]

    题意: $n*m:n,m \le 10^6$的网格,每个$2 \times 2$的方格必须有1个或3个涂成红色,其余涂成蓝色 有一些方格已经有颜色 求方案数 太神了!!!花我三节课 首先想了一下只有两 ...

  2. [BZOJ2303][Apio2011]方格染色

    [BZOJ2303][Apio2011]方格染色 试题描述 Sam和他的妹妹Sara有一个包含n × m个方格的 表格.她们想要将其的每个方格都染成红色或蓝色. 出于个人喜好,他们想要表格中每个2 × ...

  3. BZOJ2303: [Apio2011]方格染色 【并查集】

    Description Sam和他的妹妹Sara有一个包含n × m个方格的表格.她们想要将其的每个方格都染成红色或蓝色.出于个人喜好,他们想要表格中每个2 × 2的方形区域都包含奇数个(1 个或 3 ...

  4. BZOJ2303 APIO2011方格染色

    这题太神了 首先我们可以发现只有当i和j都是偶数时a[1][1]^a[1][j]^a[i][1]^a[i][j]=1才满足情况,其它时都为0 所以我们可以先把i和j都为偶数的地方^1变为0 下面才是最 ...

  5. BZOJ2303 APIO2011方格染色(并查集)

    比较难想到的是将题目中的要求看做异或.那么有ai,j^ai+1,j^ai,j+1^ai+1,j+1=1.瞎化一化可以大胆猜想得到a1,1^a1,j^ai,1^ai,j=(i-1)*(j-1)& ...

  6. BZOJ_2303_[Apio2011]方格染色 _并查集

    BZOJ_2303_[Apio2011]方格染色 _并查集 Description Sam和他的妹妹Sara有一个包含n × m个方格的 表格.她们想要将其的每个方格都染成红色或蓝色. 出于个人喜好, ...

  7. bzoj 2303: [Apio2011]方格染色【并查集】

    画图可知,每一行的状态转移到下一行只有两种:奇数列不变,偶数列^1:偶数列不变,奇数列^1 所以同一行相邻的变革染色格子要放到同一个并查集里,表示这个联通块里的列是联动的 最后统计下联通块数(不包括第 ...

  8. bzoj 2303: [Apio2011]方格染色

    传送门 Description Sam和他的妹妹Sara有一个包含n × m个方格的表格.她们想要将其的每个方格都染成红色或蓝色.出于个人喜好,他们想要表格中每个2 × 2的方形区域都包含奇数个(1 ...

  9. 【题解】P3631 [APIO2011]方格染色

    很有意思的一道题,所以单独拿出来了. 完整分享看 这里 题目链接 luogu 题意 有一个包含 \(n \times m\) 个方格的表格.要将其中的每个方格都染成红色或蓝色.表格中每个 \(2 \t ...

随机推荐

  1. 网络类型IPv4和IPv6什么意思?区别?

    在windows 7以上系统中,在设置本地IP地址的时候经常会看到同事含有IPV4协议项与IPV6协议项,并不同于以往windows xp系统中仅有TCP/IP协议项,不少朋友都觉得比较奇怪,询问编辑 ...

  2. Mysql高级查询 内连接和外连接详解

    一.内连接(INNER JOIN) 1.等值连接 概述:指使用等号"="比较两个表的连接列的值,相当于两表执行笛卡尔后,取两表连结列值相等的记录. 语法: SELECT 列 FRO ...

  3. Extmail 批量添加邮箱用户

    Extmail  设置批量添加邮箱用户 需要修改  userctl.pl  文件 修改 userctl.pl 文件 cd /var/www/extsuite/extman/tools 编辑 userc ...

  4. 虚方法virtual、抽象方法abstract、接口interface区别

    接口.抽象类.抽象方法.虚方法: 这四个名词时非常容易混淆的: 首先说一下接口 与抽象类 的异同: 相同点: 1.他们都不能实例化自己,也就是说都是用来被继承的. 2.抽象类中的抽象方法和接口方法一样 ...

  5. Java 学习札记(二)TomCat安装配置

    1.下载TomCat 下载地址:http://tomcat.apache.org/ 2.配置环境变量 CATALINA_HOME:F:\JAVA\apache-tomcat-6.0.18\apache ...

  6. Python 入门基础20 --面向对象_继承、组合

    今日内容 组合:自定义类的对象作为类的属性 继承:父类与子类.多继承 1.组合 将自定义类的对象作为类的属性 class Teacher: def __init__(self, name): self ...

  7. Docker容器命令

    ★根本前提:本地主机有镜像才能创建容器 ⒈docker run [Options] 镜像名称或镜像ID [Command] [Arg...] 用途:利用镜像创建容器实例 Options说明(常用):注 ...

  8. 【ARTS】01_11_左耳听风-20190121~20190127

    ARTS: Algrothm: leetcode算法题目 Review: 阅读并且点评一篇英文技术文章 Tip/Techni: 学习一个技术技巧 Share: 分享一篇有观点和思考的技术文章 Algo ...

  9. script & scriptreplay

    script是什么 scirpt就是一个命令,可以制作一份记录输出到终端的记录.对于那些想要真实记录终端会话的人来说,这很有用.该记录可以保存并在以后再打印出来. 怎么用 默认情况下,我们可以通过在终 ...

  10. html的结构-厂子型的布局

    上图所示的布局设计是很常见的.这个该怎么做呢? 技术需求:header 要固定住在顶部,不随鼠标滚动而向上移动:左边的div的有一定的宽度,但是要贴浏览器的底部(屏幕顶部):右边的dv要占据右边的全屏 ...