洛谷P1117 棋盘游戏

来源:互联网 发布:hexdump for vscode 编辑:程序博客网 时间:2024/06/05 11:34
洛谷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]数组代替。

0 0
原创粉丝点击