地址:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=5267

题意:Leo 有一个N*N 的格子,他又有一把魔法刷,这个刷子能把一行刷成黑色,一列刷成白色,每一行或每一列都只能刷一次。刚开始时,格子既不是白色,也不是黑色。

输入:题目给了刷好后的格子,其中'X'代表黑,’O’代表白。

输出:输出涂法。每一步用”R#”或”C#”表示,”#”指代数字,”R#”表示涂第’#行,”C#”表示涂第”#”列。输出时输出最小方案。最小方案是指:C<R,”#”也应该尽可能小。

思路:如果是一个有解的输入,那必然有一行或一列是同色的,即不过怎么涂,你最后一笔必然会使一行或一列同色。可以从这点出发,倒推,发现一行或一列同色时,使其还原为这一步之前的状态。

注意:因为是倒推,所以每一步都应优先取行,再取列,然后行号或列号应尽可能大,才能使之输出的序列最小。

关于还原上一步,如果用’-’表示没涂的状态,要考虑每个方格上一部的状态是’X’还是’O’,亦或是’-’;

举例说明:

输入     XX

OX

第一步:去第一行,

其中第一列有’O’,第二列没’O’,

所以可以化为:

O-

OX   //R1

第二步 :去第一列,

其中第一行没’X’,第二行有’X’

所以可以化为

- -

XX   //C1

第三步:去第二行,

其中第一列没’O’,第二列没’O’

所以可以化为

--

--  //R2

此时已经完成,逆序输出步骤即可

如果过程中不存在一行或一列是同色的,则可以判段此输入是非法的

/*
1500ms过的,一定还可以优化,
这次代码可能有点乱,请见谅
*/
#include <iostream>
#include <cstring>
#include <cstdio>
#include <stack>
using namespace std; typedef struct node{
char dir;
int order;
}node; //用于表示步骤,dir代表R或C,order代表行号或列号 const int maxn = 500+20;
bool row[maxn], column[maxn]; //记录此行是否涂过,记录此列是否涂过
char map[maxn][maxn]; //输入的格子图
int N; inline void input() //输入
{
cin>>N;
int i,j;
for(i=0;i<N;i++)
{
getchar();
for(j=0;j<N;j++)
scanf("%c",&map[i][j]);
//cin>>map[i][j];
}
memset(row,false,sizeof(row));
memset(column,false,sizeof(column)); //数组初始化
return ;
} int judge() //判断是否已经回到最初的状态
{
int i,j;
for(i=0;i<N;i++)
for(j=0;j<N;j++)
if( map[i][j]!='-' )
return 1;
return 0;
} inline void paint()
{
stack <node> st; //存储步骤
int i,j,k;
int flag=0; //记录有没有找到一行或一列同色
node t;
//int turn=0;
while( judge())
{
flag=0;
for(i=N-1;i>=0;i--) //行'X'
{
if(row[i])
continue;
for( j=0;j<N;j++ )
if(map[i][j]!='X') //是否有一行全部为‘X’
break;
if(j==N)
{
flag=1;
break;
}
} if(flag)
{
row[i]=true;
for(j=0;j<N;j++)
{
for(k=0;k<N;k++)
if( k!=i && map[k][j]=='O' )
break;
if( k==N )
map[i][j]='-';
else
map[i][j]='O'; //还原到上一步的状态
}
t.dir='R';
t.order=i;
st.push(t);
continue;
} flag=0;
//cout<<turn<<endl;
for(i=N-1;i>=0;i--) //列'O'
{
if(column[i])
continue;
for( j=0;j<N;j++ )
if(map[j][i]!='O') //是否有一列全部为‘O’
break;
if(j==N)
{
flag=1;
break;
}
} if(flag)
{
column[i]=true;
for(j=0;j<N;j++)
{
for(k=0;k<N;k++)
if( k!=i && map[j][k]=='X' )
break;
if( k==N )
map[j][i]='-';
else
map[j][i]='X'; //还原到上一步的状态
}
t.dir='C';
t.order=i;
st.push(t);
continue;
}
else
break; //没有找到一行或一列同色,退出
} //输出
if( judge() )
cout<<"No solution\n";
else
{
t=st.top();
st.pop();
printf("%c%d",t.dir,t.order+1);
while( !st.empty() )
{
t=st.top();
st.pop();
printf(" %c%d",t.dir,t.order+1);
}
printf("\n");
}
} int main()
{
int T;
cin>>T;
while(T--)
{
input();
paint();
}
return 0;
}

zoj p3780 Paint the Grid Again的更多相关文章

  1. ZOJ 3780 Paint the Grid Again(隐式图拓扑排序)

    Paint the Grid Again Time Limit: 2 Seconds      Memory Limit: 65536 KB Leo has a grid with N × N cel ...

  2. ZOJ 3781 Paint the Grid Reloaded(BFS+缩点思想)

    Paint the Grid Reloaded Time Limit: 2 Seconds      Memory Limit: 65536 KB Leo has a grid with N rows ...

  3. ZOJ 3781 Paint the Grid Reloaded(BFS)

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3781 Leo has a grid with N rows an ...

  4. ZOJ 3781 - Paint the Grid Reloaded - [DFS连通块缩点建图+BFS求深度][第11届浙江省赛F题]

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3781 Time Limit: 2 Seconds      Me ...

  5. ZOJ 3780 - Paint the Grid Again - [模拟][第11届浙江省赛E题]

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3780 Time Limit: 2 Seconds      Me ...

  6. ZOJ 3780 Paint the Grid Again

    拓扑排序.2014浙江省赛题. 先看行: 如果这行没有黑色,那么这个行操作肯定不操作. 如果这行全是黑色,那么看每一列,如果列上有白色,那么这一列连一条边到这一行,代表这一列画完才画那一行 如果不全是 ...

  7. ZOJ 3781 Paint the Grid Reloaded(DFS连通块缩点+BFS求最短路)

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=5268 题目大意:字符一样并且相邻的即为连通.每次可翻转一个连通块X( ...

  8. ZOJ 3781 Paint the Grid Reloaded 连通块

    LINK:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3781 题意:n*m只由OX组成的矩阵,可以选择某一连通块变成另一 ...

  9. ZOJ 3781 Paint the Grid Reloaded

    枚举,$BFS$,连通块缩点. 可以枚举一开始染哪个位置,然后逐层往外染色,看最多需要多少操作次数,也就是算最短距离.连通块缩点之后可以保证是一个黑白相间的图,且每条边的费用均为$1$,$BFS$即可 ...

随机推荐

  1. 我用过的Linux命令--修改主机名

    下面的所有设置都是在CentOS中实验的,其他的Linux系统的玩法是一样的,只是配置文件的目录可能不一样. 局域网中的主机名类似于广域网中的域名,修改主机名有两种方式: (1)在当前会话中修改主机名 ...

  2. Hibernate学习之缓存简析

    一.一级缓存 Hibernate的Session提供了一级缓存的功能,默认总是有效的,当应用程序保存持久化实体.修改持久化实体时,Session并不会立即把这种改变提交到数据库,而是缓存在当前的Ses ...

  3. new Date()的参数

    前两天发现手机页面的倒计时在Android上正常显示,在iPhone却不能显示. 后来又发现在ff和ie里也不显示.(以前只在chrome里看过,显示正常). 后来同事改了new Date()里字符串 ...

  4. 指针和引用区别 C++

    #include <iostream> using namespace std; int main(){ ; void cubeByPoint(int *);//指针传参声明 void c ...

  5. 操作 IoT 设备内嵌 SQLite

    Win10 IoT C#开发 5 - 操作 IoT 设备内嵌 SQLite 数据库 CURD   Windows 10 IoT Core 是微软针对物联网市场的一个重要产品,与以往的Windows版本 ...

  6. 软件包管理_rpm命令管理_yum工具管理_文件归档压缩_源码包管理

    rpm命令管理软件 对于挂载的像U盘那种都会在midea目录下,但是会显示在桌面上 安装软件(i:install,v:verbose冗长的,h:human):rpm  -ivh  xxxx.rpm 安 ...

  7. 如何给变量取个简短且无歧义的名字(转) good

    湾区日报上分享的一篇文章,文章的作者在Google设计Dart语言,就变量命名方面给了4点建议,文中也列出了好变量名.坏变量名的对比.不管作者的看法与你实际中的命名习惯是否一致,看完这篇文章,相信可以 ...

  8. zoj 3656 2-sat 不错的题

    http://acm.zju.edu.cn/onlinejudge/showProblem.do? problemId=4879 TLE了一下午.然后没办法了 去搜题解 发现思路跟我的差点儿相同 可是 ...

  9. 对象图(Object Diagram)—UML图(三)

    一.用一张图来介绍一下对象图的基本内容 二.对象图与类图的基本差别 三.对象图实例

  10. C#多线程编程简述

    NET将关于多线程的功能定义在System.Threading名字空间中.因此,要使用多线程,必须先声明引用此名字空间(using System.Threading;). a.启动线程顾名思义,“启动 ...