http://acm.hdu.edu.cn/showproblem.php?pid=1043

题意:给出一个八数码,求出到达指定状态的路径。

思路:路径寻找问题。在这道题里用到的知识点挺多的。第一次用双向BFS来做。

         ①双向BFS

            在单向BFS的基础上,多建一个从终止状态开始搜索的队列,当然这个时候需要两个vis[]辅助数组,分别记录两个队列的访问情况,当两个队列相遇时即可终止循环。

         ②康托展开

            X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! ,其中a[i]为当前未出现的元素中是排在第几个(从0开始)。这就是康托展开。

            例:321的康托展开为:2*2!+1*1!+0*0!;

                  (数字3后比3小的数有2个,数字2后比2小的数有1个...依次类推)

         ③逆序数判断

            每次交换两个数字逆序数的奇偶性不变,这个线代里有说过的,这样就可以根据最后状态的逆序数奇偶性来判断当前所给状态能给转换成功。不然的话好像是会超时的。

  1 #include<iostream>
  2 #include<string>
  3 #include<cstring>
  4 #include<queue>
  5 #include<algorithm>
  6 using namespace std;
  7 
  8 const int maxn = 400000;
  9 const char* DIR1 = "udlr";
 10 const char* DIR2 = "durl";
 11 
 12 int fac[] = { 1, 1, 2, 6, 24, 120, 720, 5040, 40320 };  //阶乘值
 13 
 14 int x; //x的位置
 15 
 16 int dir[] = { -3, 3, -1, 1 };
 17 
 18 int vis1[maxn];
 19 int vis2[maxn];
 20 
 21 char ss[] = "123456780";
 22 char str[30];
 23 
 24 struct Node
 25 {
 26     int x;
 27     char str[30];
 28 };
 29 
 30 struct Step
 31 {
 32     int cnt;
 33     char dir;
 34 }pre[maxn];  //记录路径
 35 
 36 
 37 int sequence(char a[])  //计算康托展开值
 38 {
 39     int sum = 0;
 40     for (int i = 0; i < 9; i++)
 41     {
 42         int k = 0;
 43         for (int j = i + 1; j < 9; j++)
 44         {
 45             if (a[j] < a[i])
 46                 k++;
 47         }
 48         sum += fac[9 - 1 - i] * k;
 49     }
 50     return sum;
 51 }
 52 
 53 
 54 void printfff(int x)  //追溯到起点输出路径
 55 {
 56     if (pre[x].cnt == -1)  return;
 57     printfff(pre[x].cnt);
 58     cout << pre[x].dir;
 59 }
 60 
 61 int judge(int x, int i)   //判断是否越界
 62 {
 63     int xx = x / 3;  //
 64     int yy = x % 3;  //
 65     if (i == 3)
 66     {
 67         int yyy = yy + 1;
 68         if (yyy > 2)  return 0;
 69     }
 70     if (i == 2)
 71     {
 72         int yyy = yy - 1;
 73         if (yyy < 0)   return 0;
 74     }
 75     if (i == 1)
 76     {
 77         int xxx = xx + 1;
 78         if (xxx>2)    return 0;
 79     }
 80     if (i == 0)
 81     {
 82         int xxx = xx - 1;
 83         if (xxx < 0) return 0;
 84     }
 85     return 1;
 86 }
 87 
 88 void bfs()
 89 {
 90     memset(vis1, 0, sizeof(vis1));
 91     memset(vis2, 0, sizeof(vis2));
 92 
 93     queue<Node> q1, q2;
 94     Node p1, p2;
 95 
 96     int count = 2;
 97     strcpy(p1.str, str);  //初始
 98     p1.x = x;             //初始x的位置
 99     //cout << p1.str << endl;
100     strcpy(p2.str, ss);   //终极
101     p2.x = 8;             //终极x的位置
102     //cout << p2.str << endl;
103     q1.push(p1);
104     q2.push(p2);
105     vis1[sequence(str)] = 1;
106     vis2[sequence(ss)] = 2;
107     pre[1].cnt = -1;  //起点标记
108     pre[2].cnt = -1;  //终点标记
109     while (!q1.empty() && !q2.empty())
110     {
111         Node u = q1.front();
112         q1.pop();
113         int p = sequence(u.str);
114         if (vis2[p])  //找到目标状态
115         {
116             printfff(vis1[p]);
117             int k = vis2[p];
118             while (pre[k].cnt != -1)
119             {
120                 cout << pre[k].dir;
121                 k = pre[k].cnt;
122             }
123             cout << endl;
124             return;
125         }
126         else //未找到目标状态
127         {
128             Node u2;
129             for (int i = 0; i < 4; i++)
130             {
131                 u2 = u;
132                 if (!judge(u.x, i))  continue;
133                 int xx = u.x + dir[i]; //x的新地址
134                 swap(u2.str[u.x], u2.str[xx]);
135                 u2.x = xx;
136                 int v = sequence(u2.str);
137                 if (vis1[v])  continue; //已访问
138                 vis1[v] = ++count;
139                 pre[count].dir = DIR1[i]; //记录方向
140                 pre[count].cnt = vis1[p];
141                 q1.push(u2);
142             }
143         }
144 
145         u = q2.front();
146         q2.pop();
147         p = sequence(u.str);
148         if (vis1[p])
149         {
150             printfff(vis1[p]);
151             int k = vis2[p];
152             while (pre[k].cnt != -1)
153             {
154                 cout << pre[k].dir;
155                 k = pre[k].cnt;
156             }
157             cout << endl;
158             return;
159         }
160         else //未找到目标状态
161         {
162             Node u2;
163             for (int i = 0; i < 4; i++)
164             {
165                 u2 = u;
166                 if (!judge(u.x, i))  continue;
167                 int xx = u.x + dir[i]; //x的新地址
168                 swap(u2.str[u.x], u2.str[xx]);
169                 u2.x = xx;
170                 int v = sequence(u2.str);
171                 if (vis2[v])  continue; //已访问
172                 vis2[v] = ++count;
173                 pre[count].dir = DIR2[i]; //记录方向
174                 pre[count].cnt = vis2[p];
175                 q2.push(u2);
176             }
177         }
178     }
179     cout << "unsolvable" << endl;
180 }
181 
182 
183 int main()
184 {
185     //freopen("D:\\txt.txt", "r", stdin);
186     char s[30];
187     while (gets(s))
188     {
189         int k = 0;
190         int t = 0;
191         for (int i = 0; s[i] != NULL; i++)
192         {
193             if (s[i] == 'x')  { str[k] = '0'; x = k; }
194             else if (s[i] >= '1' && s[i] <= '8')  str[k] = s[i];
195             else continue;
196             k++;
197         }
198         str[k] = '\0';
199         //int result = sequence(ss);
200         //cout << result << endl;
201         //cout << str << endl << ss << endl << x << endl;
202 
203         for (int i = 0; i<9; i++)  //逆序数判断是否可行
204         {
205             if (str[i] == '0')continue;
206             for (int j = 0; j<i; j++)
207             {
208                 if (str[j] == '0')continue;
209                 if (str[j]>str[i])t++;
210             }
211         }
212         if (t & 1) cout << "unsolvable" << endl;
213         else bfs();
214     }
215     return 0;
216 }