费解的开关
题目
- 你玩过“拉灯”游戏吗?25盏灯排成一个5x5的方形 。每一个灯 都有一个开关,游戏者可 以改变它的状态。每一步,游戏者可以改变某一个灯的状态。游戏者改变一个灯的 状态会产生连锁反应:和这个灯上下左右相邻 的灯也要相应地改变其状态。
我们用数字“1”表示一盏开着的灯, 用数字“0”表示关着的灯。下面这种状态
10111
01101
10111
10000
11011
01111
11101
10111
10000
11011
01111
11001
11001
10100
11011
- 给定一些游戏的初始状态,编写程序判断游戏者是否可能在6步以内使所有的灯都变亮。
输入
- 第一行有一个正整数n,代表数据中共有n个待解决的游戏初始状态。
以下若干行数据分为n组,每组数据有5行,每行5个字符。每组数据描述了一个游戏的初始状态。各组数据间用一个空行分隔。
对于30%的数据,n<=5;
对于100%的数据,n<=500。
输出
- 输出数据一共有n行,每行有一个小于等于6的整数,它表示对于输入数据中对应的游戏状态最少需要几步才能使所有灯变亮。
对于某一个游戏初始状态,若6步以内无法使所有灯变亮,请输出“-1”。
输入样例
3
00111
01011
10001
11010
11100
11101
11101
11110
11111
11111
01111
11111
11111
11111
11111
输出样例
3
2
-1
解题思路
- 其实这道题就是一道 结论+DFS,结论就是不断的固定每一行,然后就用下一行的开关来把上一行开的灯给关上,一直到最后一行,看是否有剩下的,如果有剩下的就说明这种方案不行,就继续搜下去,搜到没有并且是最优的即可.
程序如下
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
int b[10][10],a[10][10],ans,n;
int cc(int winn)
{
int s=winn;
for(int i=1;i<=5;++i)
{
for(int j=1;j<=5;++j)
{
b[i][j]=a[i][j];
}
}
for(int i=1;i<=4;++i)
{
for(int j=1;j<=5;++j)
{
if(!b[i][j])
{
s++;
b[i][j]=1-b[i][j];
b[i+1][j]=1-b[i+1][j];
b[i+1][j-1]=1-b[i+1][j-1];
b[i+1][j+1]=1-b[i+1][j+1];
b[i+2][j]=1-b[i+2][j];
}
}
}
for(int i=1;i<=5;++i)
{
if(!b[5][i]) return 10;
}
return s;
}
void dfs(int win,int sum)
{
if(win>5)
{
ans=min(ans,cc(sum));
return;
}
a[1][win]=1-a[1][win];
a[1][win-1]=1-a[1][win-1];
a[1][win+1]=1-a[1][win+1];
a[2][win]=1-a[2][win];
dfs(win+1,sum+1);
a[1][win]=1-a[1][win];
a[1][win-1]=1-a[1][win-1];
a[1][win+1]=1-a[1][win+1];
a[2][win]=1-a[2][win];
dfs(win+1,sum);
return;
}
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;++i)
{
for(int j=1;j<=5;++j)
{
for(int k=1;k<=5;++k)
{
scanf("%1d",&a[j][k]);
}
}
ans=10;
dfs(1,0);
if(ans<7) printf("%d\n",ans);
else printf("-1\n");
}
return 0;
}