Admiral(双向BFS + Hash)

Problem Description

Suppose that you are an admiral of a famous naval troop. Our naval forces have got 21 battleships. There are 6 types of battleships. First, we have got one flagship in which the admiral must be and it is denoted by number 0. Others are denoted by number from 1 to 5, each of them has 2, 3, 4, 5, 6 ships of its kind. So, we have got 21 battleships in total and we must take a giant battle against the enemy. Hence, the correct strategy of how to arrange each type of battleships is very important to us. The shape of the battlefield is like the picture that is shown below. To simplify the problem, we consider all battleships have the same rectangular shape. Fortunately, we have already known the optimal state of battleships. As you can see, the battlefield consists of 6 rows. And we have 6 types of battleship, so the optimal state is that all the battleships denoted by number i are located at the i-th row. Hence, each type of battleship corresponds to different color. You are given the initial state of battlefield as input. You can change the state of battlefield by changing the position of flagship with adjacent battleship. Two battleships are considered adjacent if and only if they are not in the same row and share parts of their edges. For example, if we denote the cell which is at i-th row and j-th position from the left as (i,j), then the cell (2,1) is adjacent to the cells (1,0), (1,1), (3,1), (3,2). Your task is to change the position of the battleships minimum times so as to reach the optimal state. Note: All the coordinates are 0-base indexed.

Input

The first line of input contains an integer T (1 <= T <= 10), the number of test cases.  Each test case consists of 6 lines. The i-th line of each test case contains i integers, denoting the type of battleships at i-th row of battlefield, from left to right.

Output

For each test case, if you can’t reach the goal in no more than 20 moves, you must output “too difficult” in one line. Otherwise, you must output the answer in one line.

SampleInput

1
1
2 0
2 1 2
3 3 3 3
4 4 4 4 4
5 5 5 5 5 5

SampleOutput

3

题意就是给你一个6*6的塔,上下两个相邻的单位可以进行交换,问最少进行几次交换,可以得到
0
1 1
2 2 2
3 3 3 3
……………………
这种状态,开始思路是用A*做,结果A*不是很熟练,没搞出来,写了个直接搜索炸了,然后我也是看了一下网上博客,使用双向搜索就行了。
思路就是从末尾开始往前搜索10步,从开始状态往后搜索10步,分别状态压缩一下存在map中,然后就看有没有两种相同的状态,否则就输出太难了。
代码:
 1 #include <bits/stdc++.h>
 2 using namespace std;
 3 #define ll long long
 4 int fx[4][2] = {1,0,1,1,-1,-1,-1,0};    //左下,右下,左上,右上
 5 
 6 struct node{
 7     ll p[6][6];
 8     int r,c;
 9     int flag;
10     int step;
11 
12     node(){}
13     node(int _r,int _c,int _flag,int _step):r(_r),c(_c),flag(_flag),step(_step){}
14 };
15 
16 queue<node>q;
17 map<ll,ll>p[2];  //分别存储两个方向的bfs状态
18 
19 ll _hash(node a){  //用hash压缩路径状态
20     ll res = 0;
21     for(int i = 0; i < 6; i++){
22         for(int j = 0; j <= i; j++){
23             res = res*6 + a.p[i][j];
24         }
25     }
26     return res;
27 }
28 
29 int bfs(node &s,node &e){
30     while(!q.empty()){
31         q.pop();
32     }
33     p[0].clear();
34     p[1].clear();
35     q.push(s);
36     q.push(e);
37     p[s.flag][_hash(s)] = 0;  //必须要标记一下,因为后面会用到count函数查询是否存在
38     p[e.flag][_hash(e)] = 0;
39     while(!q.empty()){
40         node now = q.front();
41         q.pop();
42         ll sta = _hash(now);
43         if(p[!now.flag].count(sta)){
44             int num = p[!now.flag][sta] +  now.step;
45             if(num <= 20)
46                 return num;
47             else
48                 continue;
49         }
50 
51         if(now.step >= 10)  //处理10步即可
52             continue;
53         for(int i = 0; i < 4; i++){
54             node nxt = now;
55             nxt.step++;
56             nxt.r += fx[i][0];
57             nxt.c += fx[i][1];
58             if(nxt.r < 0 || nxt.r > 6 || nxt.c < 0 || nxt.c > nxt.r)
59                 continue;
60             swap(nxt.p[now.r][now.c],nxt.p[nxt.r][nxt.c]);
61             if(p[nxt.flag].count(_hash(nxt)) == 0)
62                 p[nxt.flag][_hash(nxt)] = nxt.step;
63             q.push(nxt);
64         }
65     }
66     return -1;
67 }
68 
69 int main(){
70     int t;
71     cin>>t;
72     node s, e;
73     while(t--){
74         for(int i = 0; i < 6; i++){
75             for(int j = 0; j <= i; j++){
76                 cin>>s.p[i][j];
77                 if(s.p[i][j] == 0)
78                     s.r = i, s.c = j;
79                 e.p[i][j] = i;
80             }
81         }
82         s.flag = 0;
83         s.step = 0;
84         e = node(0,0,1,0);
85         int ans = bfs(s,e);
86         if(ans >= 0 && ans <= 20)
87             cout << ans << endl;
88         else
89             cout << "too difficult" << endl;
90     }
91     return 0;
92 }
 
   

 

 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值