Time Limit: 1000MS   Memory Limit: 10000K
Total Submissions: 8481   Accepted: 5479

Description

In an extended version of the game Lights Out, is a puzzle with 5 rows of 6 buttons each (the actual puzzle has 5 rows of 5 buttons each). Each button has a light. When a button is pressed, that button and each of its (up to four) neighbors above, below, right and left, has the state of its light reversed. (If on, the light is turned off; if off, the light is turned on.) Buttons in the corners change the state of 3 buttons; buttons on an edge change the state of 4 buttons and other buttons change the state of 5. For example, if the buttons marked X on the left below were to be pressed,the display would change to the image on the right. 

The aim of the game is, starting from any initial set of lights on in the display, to press buttons to get the display to a state where all lights are off. When adjacent buttons are pressed, the action of one button can undo the effect of another. For instance, in the display below, pressing buttons marked X in the left display results in the right display.Note that the buttons in row 2 column 3 and row 2 column 5 both change the state of the button in row 2 column 4,so that, in the end, its state is unchanged. 

Note: 
1. It does not matter what order the buttons are pressed. 
2. If a button is pressed a second time, it exactly cancels the effect of the first press, so no button ever need be pressed more than once. 
3. As illustrated in the second diagram, all the lights in the first row may be turned off, by pressing the corresponding buttons in the second row. By repeating this process in each row, all the lights in the first 
four rows may be turned out. Similarly, by pressing buttons in columns 2, 3 ?, all lights in the first 5 columns may be turned off. 
Write a program to solve the puzzle.

Input

The first line of the input is a positive integer n which is the number of puzzles that follow. Each puzzle will be five lines, each of which has six 0 or 1 separated by one or more spaces. A 0 indicates that the light is off, while a 1 indicates that the light is on initially.

Output

For each puzzle, the output consists of a line with the string: "PUZZLE #m", where m is the index of the puzzle in the input file. Following that line, is a puzzle-like display (in the same format as the input) . In this case, 1's indicate buttons that must be pressed to solve the puzzle, while 0 indicate buttons, which are not pressed. There should be exactly one space between each 0 or 1 in the output puzzle-like display.

Sample Input

2
0 1 1 0 1 0
1 0 0 1 1 1
0 0 1 0 0 1
1 0 0 1 0 1
0 1 1 1 0 0
0 0 1 0 1 0
1 0 1 0 1 1
0 0 1 0 1 1
1 0 1 1 0 0
0 1 0 1 0 0

Sample Output

PUZZLE #1
1 0 1 0 0 1
1 1 0 1 0 1
0 0 1 0 1 1
1 0 0 1 0 0
0 1 0 0 0 0
PUZZLE #2
1 0 0 1 1 1
1 1 0 0 0 0
0 0 0 1 0 0
1 1 0 1 0 1
1 0 1 1 0 1

Source

 
两种做法,一开始的做法是枚举,即枚举第0行的按不按的情况,这样第1行按的情况由第0行来决定,比如第a[0][3]=1,那么就要按[1][3]才能使得[0][3]=0,最后看第4行是不是全为0就可以了
#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std;
int a[][], rcd[][], ans[][];
int dir[][] = {{, }, {, }, {, -}, {-, }};
void push(int x, int y) {
ans[x][y] = ;
rcd[x][y] ^= ;
for(int k = ; k < ; ++k) {
int xx = x + dir[k][];
int yy = y + dir[k][];
if(xx < || xx >= || yy < || yy > ) continue;
rcd[xx][yy] ^= ;
}
}
bool check(int cur) {
for(int i = ; i < ; ++i) if(( << i) & cur) {
push(, i);
}
for(int i = ; i < ; ++i) {
for(int j = ; j < ; ++j)
if(rcd[i - ][j]) push(i, j);
}
for(int i = ; i < ; ++i) if(rcd[][i]) return false;
return true;
}
void solve() {
for(int i = ; i < ( << ); ++i) {
for(int r = ; r < ; ++r)
for(int c = ; c < ; ++c)
rcd[r][c] = a[r][c];
memset(ans, , sizeof ans);
if(check(i)) break;
}
}
void out() {
for(int i = ; i < ; ++i) {
for(int j = ; j < ; ++j)
if(j == ) printf("%d\n", ans[i][j]);
else printf("%d ", ans[i][j]);
}
}
int main() {
int _, cas = ; scanf("%d", &_);
while(_ --) {
for(int i = ; i < ; ++i)
for(int j = ; j < ; ++j)
scanf("%d", &a[i][j]);
printf("PUZZLE #%d\n", cas++);
solve();
out();
}
}

学到了另一种做法是高斯消元,可以形成(行数*列数)个方程,未知数的个数也是(行数*列数),即按下(i,j),相当于原矩阵异或x*Aij,x取0或1(不按或按)

Aij是按下位置(i,j)时所影响的位置代表的矩阵,比如3*3的矩阵,按下(1,1),那么A11 = 0 1 0, 按下(0,1), A01 = 1 1 1

                                        1 1 1           0 1 0

                                        0 1 0           0 0 0

设原矩阵为M

对于每个位置(i,j),我们考虑按或不按,那么就有M ^ x(i,j)*Aij = O, O表示0矩阵,等式两边同时异或M,那么有x(i,j)*Aij = M,两个矩阵相等,即为每个每个位置的元素对应相等,那么就可以建立(i*j)个方程组

#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std;
const int MAXN = ;
int a[MAXN][MAXN], b[MAXN][MAXN], x[MAXN];
int equ, var;
void Gauss() {
int i, j, k, col, maxr, temp;
for(i = ; i <= var; ++i) x[i] = ;
for(k = , col = ; k < equ && col < var; k++, col++) {
maxr = k;
for(int i = k + ; i < equ; ++i)
if(abs(a[i][col]) > abs(a[maxr][col]))
maxr = i; if(a[maxr][col] == ) { k--; continue; }
if(k != maxr) {
for(j = k; j < var + ; ++j)
swap(a[k][j], a[maxr][j]);
}
for(i = k + ; i < equ; ++i) {
if(a[i][col] != ) {
// LCM = lcm(abs(a[i][col]), abs(a[k][col]));
// ta = LCM / abs(a[i][col]);
// tb = LCM / abs(a[k][col]);
// if(a[i][col] * a[k][col] < 0) tb = -tb;
for(j = col; j < var + ; ++j)
a[i][j] = a[i][j] ^ a[k][j];
}
}
}
for(i = var - ; i >= ; --i) {
temp = a[i][var];
for(j = i + ; j < var; ++j)
temp ^= (a[i][j] * x[j]);
x[i] = temp;
}
}
int vis[][];
int dir[][] = {{, }, {, }, {-, }, {, -} };
void get(int x, int y)
memset(vis, , sizeof vis);
vis[x][y] = ;
for(int i = ; i < ; ++i) {
int xx = x + dir[i][];
int yy = y + dir[i][];
if(xx < || xx >= || yy < || yy > ) continue;
vis[xx][yy] = ;
}
for(int i = ; i < ; ++i) {
for(int j = ; j <= ; ++j) printf("%d ", vis[i][j]);
puts("");
}
puts(""); void debug() {
for(int i = ; i < ; ++i) {
for(int j = ; j <= ; ++j)
printf("%d ", a[i][j]);
puts("");
}
}
void init() {
memset(a, , sizeof a);
equ = var = ;
int cur = ;
for(int i = ; i < ; ++i)
for(int j = ; j <= ; ++j)
a[cur++][] = b[i][j];
// debug();
cur = ;
for(int i = ; i < ; ++i) {
for(int j = ; j <= ; ++j) {
get(i, j);
int k = ;
for(int r = ; r < ; ++r)
for(int c = ; c <= ; ++c)
a[k++][cur] = vis[r][c];
cur++;
}
}
// debug();
} void out() {
for(int i = ; i < ; ++i) {
if((i + ) % == ) printf("%d\n", x[i]);
else printf("%d ", x[i]);
}
}
int main() {
int _, cas = ; scanf("%d", &_);
while(_ --) {
for(int i = ; i < ; ++i)
for(int j = ; j <= ; ++j)
scanf("%d", &b[i][j]);
init();
printf("PUZZLE #%d\n", cas++);
Gauss();
// debug();
out();
}
}

poj1222 EXTENDED LIGHTS OUT 高斯消元||枚举的更多相关文章

  1. POJ1222 EXTENDED LIGHTS OUT 高斯消元 XOR方程组

    http://poj.org/problem?id=1222 在学校oj用搜索写了一次,这次写高斯消元,haoi现场裸xor方程消元没写出来,真实zz. #include<iostream> ...

  2. [poj1222]EXTENDED LIGHTS OUT(高斯消元)

    题意:每个灯开启会使自身和周围的灯反转,要使全图的灯灭掉,判断灯开的位置. 解题关键:二进制高斯消元模板题. 复杂度:$O({n^3})$ #include<cstdio> #includ ...

  3. POJ 1222 EXTENDED LIGHTS OUT (高斯消元)

    题目链接 题意:5*6矩阵中有30个灯,操作一个灯,周围的上下左右四个灯会发生相应变化 即由灭变亮,由亮变灭,如何操作使灯全灭? 题解:这个问题是很经典的高斯消元问题.同一个按钮最多只能被按一次,因为 ...

  4. EXTENDED LIGHTS OUT (高斯消元)

    In an extended version of the game Lights Out, is a puzzle with 5 rows of 6 buttons each (the actual ...

  5. POJ 1222 EXTENDED LIGHTS OUT [高斯消元XOR]

    题意: $5*6$网格里有一些灯告诉你一开始开关状态,按一盏灯会改变它及其上下左右的状态,问最后全熄灭需要按那些灯,保证有解 经典问题 一盏灯最多会被按一次,并且有很明显的异或性质 一个灯作为一个方程 ...

  6. POJ 3185 The Water Bowls(高斯消元-枚举变元个数)

    题目链接:http://poj.org/problem?id=3185 题意:20盏灯排成一排.操作第i盏灯的时候,i-1和i+1盏灯的状态均会改变.给定初始状态,问最少操作多少盏灯使得所有灯的状态最 ...

  7. BZOJ 1770: [Usaco2009 Nov]lights 燈( 高斯消元 )

    高斯消元解xor方程组...暴搜自由元+最优性剪枝 -------------------------------------------------------------------------- ...

  8. POJ 1681 Painter's Problem(高斯消元+枚举自由变元)

    http://poj.org/problem?id=1681 题意:有一块只有黄白颜色的n*n的板子,每次刷一块格子时,上下左右都会改变颜色,求最少刷几次可以使得全部变成黄色. 思路: 这道题目也就是 ...

  9. BZOJ1770:[USACO]lights 燈(高斯消元,DFS)

    Description 貝希和她的閨密們在她們的牛棚中玩遊戲.但是天不從人願,突然,牛棚的電源跳閘了,所有的燈都被關閉了.貝希是一個很膽小的女生,在伸手不見拇指的無盡的黑暗中,她感到驚恐,痛苦與絕望. ...

随机推荐

  1. 容器配置jndi Tomcat为例

    context.xml 文件 <Resource name=" password="spsj" type="javax.sql.DataSource&qu ...

  2. 合唱队形2(洛谷U5874)

    题目背景 上次老师挑出来的(N-K)位同学很不高兴,于是他们准备自己组建合唱队形.他们请了kkk来帮忙. 题目描述 他们安排了一个动作--手拉着手唱一首歌(就是他们围成一个圈).如果有两个相邻的同学的 ...

  3. 借教室(codevs 1217)

    1217 借教室 2012年NOIP全国联赛提高组  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond 题解  查看运行结果     题目描述 Descrip ...

  4. instanceof、 isinstance 与 isAssignableFrom的区别

    instanceof运算符 只被用于对象引用变量,检查左边的被测试对象 是不是 右边类或接口的 实例化.如果被测对象是null值,则测试结果总是false. 形象地:自身实例或子类实例 instanc ...

  5. XPath的基本使用

    XPath XPath 使用路径表达式来选取 XML 文档中的节点或节点集. 路径表达式 结果 bookstore 选取 bookstore 元素的所有子节点. /bookstore 选取根元素 bo ...

  6. HBase参数配置及说明(转)

    版本:0.94-cdh4.2.1 hbase-site.xml配置 hbase.tmp.dir 本地文件系统tmp目录,一般配置成local模式的设置一下,但是最好还是需要设置一下,因为很多文件都会默 ...

  7. Hadoop家族 路线图(转)

    主要介绍Hadoop家族产品,常用的项目包括Hadoop, Hive, Pig, HBase, Sqoop, Mahout, Zookeeper, Avro, Ambari, Chukwa,新增加的项 ...

  8. JavaScript基础——添加错误处理

    JavaScript编程的一个重要组成部分,是添加错误处理来应对可能会出现的问题.默认情况下,如果因为你的JavaScript中的问题二产生了一个代码异常,那么脚本就会失败并且无法完成加载.这通常不是 ...

  9. Spring学习笔记—Spring之旅

    1.Spring简介     Spring是一个开源框架,最早由Rod Johnson创建,并在<Expert One-on-One:J2EE Design and Development> ...

  10. Newtonsoft.Json(Json.Net)学习笔记-高级使用(转)

    1.忽略某些属性 2.默认值的处理 3.空值的处理 4.支持非公共成员 5.日期处理 6.自定义序列化的字段名称 7.动态决定属性是否序列化 8.枚举值的自定义格式化问题 9.自定义类型转换 10.全 ...