启发式搜索算法求解八数码问题(C)


下午看一个游戏的算法时看了一下启发式搜索算法,心血来潮跑了一遍很久很久以前写八数码的程序(C语言),发现各种问题,后来顺着思路整理了一下,贴出来和大家分享一下,直接上代码:

//
//  main.c
//  yunsuan
//
//  Created by mac on 12-8-7.
//  Copyright 2012年 __MyCompanyName__. All rights reserved.
//

#include 
#include 
#include 

#define N 3 //数码组大小
#define Max_Step 50 //最大搜索深度
#define MAX 50

typedef struct node//八数码结构体
{
    int form[N][N];//数码组
    int evalue;//评估值
    int udirect;//所屏蔽方向,防止往回推到上已状态,1上2下3左4右
    struct node *parent;//父节点
}Graph;

Graph *Qu[MAX];//队列
Graph *St[MAX];//堆栈

/////////打印数码组
void Print(Graph *The_graph)
{
    int i,j;
    if(The_graph==NULL)
        printf("图为空\n");
    else
    {
        printf("---------------------\n");
        for(i=0;iform[i][j]);//遍历打印
            }
            printf("\t|\n");
        }
        printf("|\t\t\t差距:%d\t|\n",The_graph->evalue);//差距显示
        printf("---------------------\n");
    }
}

/////////评价函数
int Evaluate(Graph *The_graph,Graph *End_graph)
{
    int valute=0;//差距数
    int i,j;
    for(i=0;iform[i][j]!=End_graph->form[i][j])
            {
                valute++;
            }
        }
    }
    The_graph->evalue=valute;
    return valute;
}

/////////移动数码组
Graph *Move(Graph *The_graph,int Direct,int CreatNew_graph)
{
    Graph *New_graph;//
    int HasGetBlank=0;//是否获取空格位置
    int AbleMove=1;//是否可移动
    int i,j,t_i,t_j,x,y;
    
    for(i=0;iform[i][j]==0)
            {
                HasGetBlank=1;
                break;
            }
        }
        if(HasGetBlank==1)
            break;
    }
    //printf("空格位置:%d,%d\n",i,j);
    
    t_i=i;
    t_j=j;
    
    //移动空格
    switch(Direct)
    {
        case 1://上
            t_i--;
            if(t_i<0)
                AbleMove=0;
            break;
        case 2://下
            t_i++;
            if(t_i>=N)
                AbleMove=0;
            break;
        case 3://左
            t_j--;
            if(t_j<0)
                AbleMove=0;
            break;
        case 4://右
            t_j++;
            if(t_j>=N)
                AbleMove=0;
            break;
            
    }
    
    if(AbleMove==0)//不能移动则返回原节点
    {
        return The_graph;
    }
    if(CreatNew_graph==1)
    {
        New_graph=(Graph *)malloc(sizeof(Graph));//生成节点
        for(x=0;xform[x][y]=The_graph->form[x][y];//复制数码组
            }

        }
    }
    else
    {
        New_graph=The_graph;
    }
    //移动后
    New_graph->form[i][j]=New_graph->form[t_i][t_j];
    New_graph->form[t_i][t_j]=0;
    //printf("移动产生的新图:\n");
    //Print(New_graph);
    return New_graph;
}

/////////搜索函数
Graph *Search(Graph *Begin,Graph *End)
{
    Graph *g1,*g2,*g;
    int Step=0;//深度
    int Direct=0;//方向
    int i;
    int front,rear;
    front=rear=-1;//队列初始化
    g=NULL;
    rear++;//入队
    Qu[rear]=Begin;
    while(rear!=front)//队列不空
    {
        front++;//出队
        g1=Qu[front];
        //printf("开始第%d个图:\n",front);
        //Print(g1);
        for(i=1;i<=4;i++)//分别从四个方向推导出新子节点
        {
            Direct=i;
            if(Direct==g1->udirect)//跳过屏蔽方向
                continue;
            g2=Move(g1, Direct, 1);//移动数码组
            
            if(g2!=g1)//数码组是否可以移动
            {
                //可以移动
                Evaluate(g2, End);//评价新的节点
                //printf("开始产生的第%d个图:\n",i);
                //Print(g2);
                if(g2->evalue<=g1->evalue+1)
                {
                    //是优越节点
                    g2->parent=g1;
                    //移动空格
                    switch(Direct)//设置屏蔽方向,防止往回推
                    {
                        case 1://上
                            g2->udirect=2;
                            break;
                        case 2://下
                            g2->udirect=1;
                            break;
                        case 3://左
                            g2->udirect=4;
                            break;
                        case 4://右
                            g2->udirect=3;
                            break;
                            
                    }
                    rear++;
                    Qu[rear]=g2;//存储节点到待处理队列
                    if(g2->evalue==0)//为0则搜索完成
                    {
                        g=g2;
                        //i=5;
                        break;
                    }
                }
                else
                {
                    free(g2);//抛弃劣质节点
                    g2=NULL;
                }
            }
            
        }
        
        if(g!=NULL)//为0则搜索完成
        {
            if(g->evalue==0)
            {
                break;
            }
        }
        
        Step++;//统计深度
        if(Step>Max_Step)
        {
            break;
        }
    }
    return g;
}

/////////初始化一个八数码结构体
Graph *CR_BeginGraph(Graph *The_graph)
{
    srand((unsigned)time(0));
    int M=10;//随机移动步数
    int Direct;
    int i,x,y;
    Graph *New_graph;
    New_graph=(Graph *)malloc(sizeof(Graph));
    for(x=0;xform[x][y]=The_graph->form[x][y];
        }
    }
    for(i=0;ievalue=0;
    New_graph->udirect=0;
    New_graph->parent=NULL;
    
    return New_graph;
}

int main (int argc, const char * argv[])
{
    
    // insert code here...
    /*
    Graph Begin_graph={
        {{2,8,3},{1,6,4},{0,7,5}},0,0,NULL
    };
    
    Graph Begin_graph={
        {{2,8,3},{1,0,4},{7,6,5}},0,0,NULL
    };
    
    
    Graph Begin_graph={
        {{2,0,1},{4,6,5},{3,7,8}},0,0,NULL
    };
    */
    
    //目标数码组
    Graph End_graph={
        {{1,2,3},{8,0,4},{7,6,5}},0,0,NULL
    };
    
    //初始数码组
    Graph *Begin_graph;
    Begin_graph=CR_BeginGraph(&End_graph);//随机产生初始数码组
    Evaluate(Begin_graph, &End_graph);//对初始的数码组评价
    printf("初始数码组:\n");
    Print(Begin_graph);
    printf("目标数码组:\n");
    Print(&End_graph);
    
    Graph *G,*P;
    int top=-1;
    
    //图搜索
    G=Search(Begin_graph, &End_graph);
    //打印
    if(G)
    {
        //把路径倒序
        P=G;
        //压栈
        while(P!=NULL)
        {
            top++;
            St[top]=P;
            P=P->parent;
        }
        printf("<<<<<<<<<<<<<<<搜索结果>>>>>>>>>>>>>>>>\n");
        //弹栈打印
        while(top>-1)
        {
            P=St[top];
            top--;
            Print(P);
        }
        printf("<<<<<<<<<<<<<<<<<完成>>>>>>>>>>>>>>>>>>\n");
    }
    else
    {
        printf("搜索不到结果,深度为%d\n",Max_Step);
        //设计搜索深度范围主要是防止队列内存越界
    }

    
    return 0;
}

 

你可能感兴趣的:(算法)