题目链接:戳我

【问题描述】

小A在玩打地鼠游戏。有一个n×m的网格,每个位置上地鼠都会要么冒出头要么缩进去。地鼠很狡猾,每次小A选一个地鼠冒出头的格子(x,y)把它打下去,但同一行同一列的地鼠全都会冒出头来。

小A发现这个游戏好像怎么都玩不完。这个时候小B过来向他展示真正的技术了。小B当然也知道这游戏是玩不完的,于是他准备了两个状态,并准备向小A表演把状态1进行若干次打地鼠操作变成状态2。

现在小B想知道他会不会翻车。

【输入格式】

第一行两个整数n,m表示棋盘大小n行m列。

接下来n行,每行一个长为m的字符串描述初始状态,'O'表示地鼠冒出头来,'X'表示地鼠缩了进去。

接下来n行,每行一个长为m的字符串描述结束状态,格式同上。

【输出格式】

如果能从初始状态变成目标状态输出1,否则输出0。

【样例输入】

4 4

XOOO

XXXX

XOOX

XOXO

OXOO

XOOO

XOOO

OOOO

【样例输出】

1

【数据规模】

subtask1(20'):n,m≤4。

subtask2(30'):n,m≤50。

subtask3(50'):n,m≤1000。


用黑色表示缩下去,白色表示冒出头。

每次操作相当于选一个白的变成黑的,但这行这列都会变成白的。

考虑倒着做。用灰色表示可能是黑也可能是白。

那么操作就变成了:选一个黑色或灰色的,必须满足这行这列除了它没有黑色,把它变成白色并把这行这列变成灰色。

注意到要操作一个黑色时,这行这列除了它就没有黑色了,所以操作之间不会干扰,直接能做就做就行了。

最后判下灰色的行列的交界处必须至少有一个白色(要进行第一步操作)。其它位置必须对应相等。

还要特判下如果一开始两个状态就相等输出1。

复杂度O(nm)。

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<queue>
#define mp make_pair
#define MAXN 1010
using namespace std;
int n,m;
int a[MAXN][MAXN],b[MAXN][MAXN],cnt_hang[MAXN],cnt_lie[MAXN];
int done[MAXN][MAXN],done_hang[MAXN],done_lie[MAXN];
char s[MAXN][MAXN],t[MAXN][MAXN];
queue<pair<int,int> >q;
//倒着做 相当于把0变成1 QAQ
inline bool check_the_same()
{
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
if(a[i][j]!=b[i][j]) return false;
return true;
}
inline bool check_s()
{
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
if(a[i][j]!=0) return false;
return true;
}
inline bool check_t()
{
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
if(b[i][j]!=1) return false;
return true;
}
inline void paint(int x,int y)
{
if(!done_hang[x])
{
done_hang[x]=1;
for(int j=1;j<=m;j++)
{
if(j==y) continue;
b[x][j]=-1;
if(!cnt_lie[j]) q.push(mp(x,j));
}
}
if(!done_lie[y])
{
done_lie[y]=1;
for(int i=1;i<=n;i++)
{
if(i==x) continue;
b[i][y]=-1;
if(!cnt_hang[i]) q.push(mp(i,y));
}
}
}
inline bool solve()
{
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
if(b[i][j]==0&&cnt_hang[i]==1&&cnt_lie[j]==1)
{
q.push(mp(i,j));
b[i][j]=-1;
done[i][j]=1;
}
while(!q.empty())
{
int u_x=q.front().first;
int u_y=q.front().second;
q.pop();
paint(u_x,u_y);
}
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
if(b[i][j]!=-1&&a[i][j]!=b[i][j])
return false;
return true;
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("ce.in","r",stdin);
#endif
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++) scanf("%s",s[i]+1);
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
{
if(s[i][j]=='O') a[i][j]=1;
else a[i][j]=0;
}
for(int i=1;i<=n;i++) scanf("%s",t[i]+1);
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
{
if(t[i][j]=='O') b[i][j]=1;
else b[i][j]=0,cnt_hang[i]++,cnt_lie[j]++;
}
if(check_the_same()) printf("1\n");
else if(check_s()||check_t()) printf("0\n");
else if(solve()) printf("1\n");
else printf("0\n");
return 0;
}

noi.ac #536 打地鼠的更多相关文章

  1. # NOI.AC省选赛 第五场T1 子集,与&最大值

    NOI.AC省选赛 第五场T1 A. Mas的童年 题目链接 http://noi.ac/problem/309 思路 0x00 \(n^2\)的暴力挺简单的. ans=max(ans,xor[j-1 ...

  2. NOI.ac #31 MST DP、哈希

    题目传送门:http://noi.ac/problem/31 一道思路好题考虑模拟$Kruskal$的加边方式,然后能够发现非最小生成树边只能在一个已经由边权更小的边连成的连通块中,而树边一定会让两个 ...

  3. NOI.AC NOIP模拟赛 第五场 游记

    NOI.AC NOIP模拟赛 第五场 游记 count 题目大意: 长度为\(n+1(n\le10^5)\)的序列\(A\),其中的每个数都是不大于\(n\)的正整数,且\(n\)以内每个正整数至少出 ...

  4. NOI.AC NOIP模拟赛 第六场 游记

    NOI.AC NOIP模拟赛 第六场 游记 queen 题目大意: 在一个\(n\times n(n\le10^5)\)的棋盘上,放有\(m(m\le10^5)\)个皇后,其中每一个皇后都可以向上.下 ...

  5. NOI.AC NOIP模拟赛 第二场 补记

    NOI.AC NOIP模拟赛 第二场 补记 palindrome 题目大意: 同[CEOI2017]Palindromic Partitions string 同[TC11326]Impossible ...

  6. NOI.AC NOIP模拟赛 第一场 补记

    NOI.AC NOIP模拟赛 第一场 补记 candy 题目大意: 有两个超市,每个超市有\(n(n\le10^5)\)个糖,每个糖\(W\)元.每颗糖有一个愉悦度,其中,第一家商店中的第\(i\)颗 ...

  7. NOI.AC NOIP模拟赛 第四场 补记

    NOI.AC NOIP模拟赛 第四场 补记 子图 题目大意: 一张\(n(n\le5\times10^5)\)个点,\(m(m\le5\times10^5)\)条边的无向图.删去第\(i\)条边需要\ ...

  8. NOI.AC NOIP模拟赛 第三场 补记

    NOI.AC NOIP模拟赛 第三场 补记 列队 题目大意: 给定一个\(n\times m(n,m\le1000)\)的矩阵,每个格子上有一个数\(w_{i,j}\).保证\(w_{i,j}\)互不 ...

  9. NOI.AC WC模拟赛

    4C(容斥) http://noi.ac/contest/56/problem/25 同时交换一行或一列对答案显然没有影响,于是将行列均从大到小排序,每次处理限制相同的一段行列(呈一个L形). 问题变 ...

随机推荐

  1. 利用requests提交相同名称数据的处理方法

    #字典键不能重复 data={ boardid' : boardid, 'divids[ ]' : '0' , 'divids[ ]' : '1' , 'divids[ ]' : '2' , } #这 ...

  2. Java反射理解(四)-- 获取成员变量构造函数信息

    Java反射理解(四)-- 获取成员变量构造函数信息 步骤 获取成员变量信息: obj.getClass() 获取类类型对象 成员变量也是对象,java.lang.reflect.Field 类中封装 ...

  3. 随便----js参考书

    一.Javascript方面的书籍: 1 JavaScript权威指南(第6版):号称javascript圣经,前端必备:前端程序员学习核心JavaScript语言和由Web浏览器定义的JavaScr ...

  4. jdbc原生操作数据库

    jdbc原生操作数据库流程: 第一步:Class.forName()加载数据库连接驱动: 第二步:DriverManager.getConnection()获取数据连接对象; 第三步:根据 SQL 获 ...

  5. Java对象的序列化和反序列化介绍

    一.什么序列化和反序列化以及作用: java序列化是指把java对象转换为字节序列的过程,而java反序列化是指把字节序列恢复为java对象的过程 1.序列化: 1)对象序列化的最主要的用处就是在传递 ...

  6. Redis之过期策略

    一.设置过期时间 Redis对存储值的过期处理实际上是针对该值的键(key)处理的,即时间的设置也是设置key的有效时间.Expires字典保存了所有键的过期时间,Expires也被称为过期字段. e ...

  7. js之数据类型(对象类型——构造器对象——数组1)

    数组是值的有序集合,每个值叫做一个元素,而每一个元素在数组中有一个位置,以数字表示,称为索引.JavaScript数组是无类型的,数组元素可以是任意类型且同一个数组中不同元素也可能有不同的类型.数组的 ...

  8. 关于学习电信nb-iot的小结

    关于这几天对nb-iot的学习的总结和遇到的坑 初步学习nb-iot,了解到了nb-iot对于传感器数据传输功能的强大: 废话不多说,对于nb-iot我们选择的有人的模块,选择B5频段也就是电信的nb ...

  9. Java高并发程序设计学习笔记(五):JDK并发包(各种同步控制工具的使用、并发容器及典型源码分析(Hashmap等))

    转自:https://blog.csdn.net/dataiyangu/article/details/86491786#2__696 1. 各种同步控制工具的使用1.1. ReentrantLock ...

  10. linux重装docker-compose后无法执行docker-compose命令

    背景 使用自动化脚本重装docker和docker-compose(但脚本中未对旧版本的docker-compose进行任何处理,比如卸载删除) 导致执行docker-compose命令时报了错,大多 ...