山东济南彤昌机械科技有限公司 山东济南江鹏工贸游有限公司

洛谷P1117 棋盘游戏

洛谷1117 棋盘游戏

题目描述

在一个4*4的棋盘上有8个黑棋和8个白棋,当且仅当两个格子有公共边,这两个格子上的棋是相邻的。移动棋子的规则是交换相邻两个棋子。现在给出一个初始棋盘和一个最终棋盘,要求你找出一个最短的移动序列使初始棋盘变为最终棋盘。
Klux说:“这么简单的题目,我都会做!”

输入输出格式

输入格式:

第1到4行每行四个数字(1或者0),描述了初始棋盘
接着是一个空行
第6到9行每行四个数字,描述了最终棋盘

输出格式:

输出文件的只有一行是一个整数n,表示最少的移动步数。

输入输出样例

输入样例#1:

1111

0000

1110

0010

 

1010

0101

1010

0101

输出样例#1:

4

 

【思路】

  宽搜+位运算。

  由于要求最小步数可以看出BFS的基本框架,但是如果用矩阵存储状态的话太耗费空间而且很慢,注意到每个格子的状态非0即1而且总格子数目为16所以可以用二进制的方法存储状态,相应判断,转移,判重。

  详见代码。

 

【代码】

 

 1 #include<iostream>
 2 #include<queue>
 3 #define FOR(a,b,c) for(int a=(b);a<(c);a++)
 4 using namespace std;
 5 
 6 const int maxn = 16;
 7 struct Node{
 8     int num,d;
 9 };
10 int A,B;
11 int vis[100000];
12 
13 
14 void BFS() {
15 queue<Node> q;
16     q.push((Node){A,0});
17     while(!q.empty()) {
18         Node u=q.front(); q.pop();
19         int tmp=u.num;
20         if(tmp==B) { cout<<u.d; return ; }
21         for(int i=15;i>=0;i--) 
22         {
23             int x=(15-i)/4,y=(15-i)%4,w=1<<i,z;
24             if(y<3 && (tmp&(1<<i))!=(tmp&(1<<i-1))) {
25                 z=1<<i-1;
26                 if(!vis[tmp^z^w]) {
27                     vis[tmp^z^w]=1;
28                     q.push((Node){tmp^z^w,u.d+1});
29                 }
30             }
31             if(x<3 && (tmp&(1<<i))!=(tmp&(1<<i-4))) {
32                 z=1<<i-4;
33                 if(!vis[tmp^z^w]) {
34                     vis[tmp^z^w]=1;
35                     q.push((Node){tmp^z^w,u.d+1});
36                 }
37             }
38         }
39     }
40 }
41 
42 int main() {
43     ios::sync_with_stdio(false);
44     char c;
45     for(int i=15;i>=0;i--) {
46         cin>>c;
47         if(c!='0')  A += 1<<i;
48     }
49     for(int i=15;i>=0;i--) {
50         cin>>c;
51         if(c!='0')  B += 1<<i;
52     }
53     if(A==B) cout<<0;
54     else BFS();
55 }

 

 

另外STL的queue一定程度上的减慢速度,可以用d[][2]数组代替。

posted on 2015-10-08 11:11  hahalidaxin  阅读(754)  评论(0编辑  收藏  举报