zoukankan      html  css  js  c++  java
  • 【算法】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[30];  
        int state[9], goal[9] = {1,2,3,4,5,6,7,8,0};  
        int dir[4][2] = {{-1,0},{1,0},{0,-1},{0,1}}; // 上,下,左, 右  
        char path_dir[5] = "udlr";  
        int st[MAXN][9];  
        int father[MAXN], path[MAXN]; // 保存打印路径  
          
        const int MAXHASHSIZE = 1000003;  
        int head[MAXHASHSIZE], next[MAXN];  
          
        void init_lookup_table() { memset(head, 0, sizeof(head)); }  
          
        typedef int State[9];  
        int hash(State& s) {  
          int v = 0;  
          for(int i = 0; i < 9; i++) v = v * 10 + 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])) == 0) return 0;  
            u = next[u];  
          }  
          next[s] = head[h];  
          head[h] = s;  
          return 1;  
        }  
          
        int bfs(){  
            init_lookup_table();  
            father[0] = path[0] = -1;  
            int front=0, rear=1;  
            memcpy(st[0], state, sizeof(state));  
              
            while(front < rear){  
                int *s = st[front];  
                 
                if(memcmp(s, goal, sizeof(goal))==0){  
                    return front;  
                }  
          
                int j;  
                for(j=0; j<9; ++j) if(!s[j])break; // 找出0的位置  
                int x=j/3, y=j%3;     // 转换成行,列  
                  
                for(int i=0; i<4; ++i){  
          
                    int dx = x+dir[i][0]; // 新状态的行,列  
                    int dy = y+dir[i][1];  
                    int pos = dx*3+dy;    // 目标的位置  
          
                    if(dx>=0 && dx<3 && dy>=0 && dy<3){  
                        int *newState = st[rear];  
                        memcpy(newState, s, sizeof(int)*9);  
                        newState[j] = s[pos];  
                        newState[pos] = 0;  
                        if(try_to_insert(rear)){  
                            father[rear] = front;  path[rear] = i;  
                            rear++;  
                        }  
                    }  
                }   
                front++;  
            }  
            return -1;  
        }  
          
        void print_path(int cur){  
            if(cur!=0){  
                print_path(father[cur]);  
                printf("%c", path_dir[path[cur]]);  
            }  
        }  
          
        int main(){  
              
            while(gets(input)){  
                // 转换成状态数组, 'x'用0代替  
                for(int pos=0, i=0; i<strlen(input); ++i){  
                    if(input[i]>='0' && input[i]<='9')  
                        state[pos++] = input[i]-'0';  
                    else if(input[i]=='x')  
                        state[pos++] = 0;  
                }  
                int ans;  
                if((ans=bfs())!=-1){   
                    print_path(ans);  
                    printf("
    ");  
                }  
            }          
        }  
  • 相关阅读:
    ansible笔记(11):初识ansible playbook(二)
    Linux下查看占用CPU与内存最高的进程
    ansible笔记(10):初识ansible playbook
    AbpZero Http 模式下 Chrome浏览器因Cookie 不能登录
    Tomcat 8443&8080 并存
    接入腾讯cos文件存储
    安卓包打渠道标签
    java Android与PHP encode的区别
    thinkphp常用
    phalcon task任务
  • 原文地址:https://www.cnblogs.com/KID-XiaoYuan/p/6412837.html
Copyright © 2011-2022 走看看