【算法】BFS+哈希解决八数码问题

时间:2023-03-09 19:24:14
【算法】BFS+哈希解决八数码问题

15拼图已经有超过100年; 即使你不叫这个名字知道的话,你已经看到了。它被构造成具有15滑动砖,每一个从1到15上,并且所有包装成4乘4帧与一个瓦块丢失。让我们把丢失的瓷砖“X”; 拼图的目的是安排瓷砖以便它们排序为:

1 2 3 4 
5 6 7 8
9 10 11 12
13 14 15×

这里唯一合法经营是交流'X'与它共享一个边缘的瓷砖之一。作为一个例子,举动下列顺序解决了一个稍微加扰难题:

1 2 3 4 1 2 3 4 1 2 3 4 1 2 3 4 
5 6 7 8 5 6 7 8 5 6 7 8 5 6 7 8
9×10 12 9 10×12 9 10 11 12 9 10 11 12
13 14 11 15 13 14 11 15 13 14×15 13 14 15×
R-> D-> R->

上一行中的字母指示哪个的“x”瓦片的邻居交换在每一步的“x”瓦片; 合法的值是'R','L','U'和'D',右,左,上,下,分别。

并非所有的难题都可以解决; 在1870年,一个叫萨姆·劳埃德的人是著名的分配难题的一个无法解决的版本,

折腾了不少人。事实上,所有你必须做的,使一个普通的益智成无法解决的一个是交换两个瓷砖(不包括课程的缺失'X'瓦)。

在这个问题中,你会写三个解决不太知名的8益智,砖组成一个三的程序

安排。


思路分析:

从第一步开始,我们每一步都有不多于四种选择:将白格向上移;向下移;向左移;向右移。这很像是做迷宫。走迷宫的任务是走到某一点就算赢。而八数码,是走到什么局面才算赢。那么队列中存放的就不是点,而是面。换句话说,就是图。现在问题变得很简单,只要你能在一张3×3的图上裸着BFS一遍,搜到目标图就算你赢了。就像迷宫的vis数组一样,我们需要对走过的局面设置标记,不要重复走。

在做这道题中学到的几样小技巧:

1. 数组直接用memcpy, memcmp对整块内存进行复制或者比较, 速度比用for循环快。

2.用typedef来定义一个新名称可以更加方便。

3.哈希表与编码的应用

    #include<stdio.h>
#include<string.h>
#define MAXN 500000 char input[];
int state[], goal[] = {,,,,,,,,};
int dir[][] = {{-,},{,},{,-},{,}}; // 上,下,左, 右
char path_dir[] = "udlr";
int st[MAXN][];
int father[MAXN], path[MAXN]; // 保存打印路径 const int MAXHASHSIZE = ;
int head[MAXHASHSIZE], next[MAXN]; void init_lookup_table() { memset(head, , sizeof(head)); } typedef int State[];
int hash(State& s) {
int v = ;
for(int i = ; i < ; i++) v = v * + s[i];
return v % MAXHASHSIZE; } int try_to_insert(int s) {
int h = hash(st[s]);
int u = head[h];
while(u) {
if(memcmp(st[u], st[s], sizeof(st[s])) == ) return ;
u = next[u];
}
next[s] = head[h];
head[h] = s;
return ;
} int bfs(){
init_lookup_table();
father[] = path[] = -;
int front=, rear=;
memcpy(st[], state, sizeof(state)); while(front < rear){
int *s = st[front]; if(memcmp(s, goal, sizeof(goal))==){
return front;
} int j;
for(j=; j<; ++j) if(!s[j])break; // 找出0的位置
int x=j/, y=j%; // 转换成行,列 for(int i=; i<; ++i){ int dx = x+dir[i][]; // 新状态的行,列
int dy = y+dir[i][];
int pos = dx*+dy; // 目标的位置 if(dx>= && dx< && dy>= && dy<){
int *newState = st[rear];
memcpy(newState, s, sizeof(int)*);
newState[j] = s[pos];
newState[pos] = ;
if(try_to_insert(rear)){
father[rear] = front; path[rear] = i;
rear++;
}
}
}
front++;
}
return -;
} void print_path(int cur){
if(cur!=){
print_path(father[cur]);
printf("%c", path_dir[path[cur]]);
}
} int main(){ while(gets(input)){
// 转换成状态数组, 'x'用0代替
for(int pos=, i=; i<strlen(input); ++i){
if(input[i]>='' && input[i]<='')
state[pos++] = input[i]-'';
else if(input[i]=='x')
state[pos++] = ;
}
int ans;
if((ans=bfs())!=-){
print_path(ans);
printf("\n");
}
}
}