C语言实现迷宫求解

最近做了一个用C语言迷宫求解的题目,来分享一下。

题目要求://迷宫的布局放入到一个二维的数组中 1代表该地方走不通为墙,0代表该地方可以走通,打印出来走的顺序
 //0 , 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9
const int mizu[10][10] = {      1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1,  //0
 1 , 0 , 0 , 1 , 0 , 0 , 0 , 1 , 0 , 1,  //1
 1 , 0 , 0 , 1 , 0 , 0 , 0 , 1 , 0 , 1,  //2
 1 , 0 , 0 , 0 , 0 , 1 , 1 , 0 , 0 , 1,  //3
 1 , 0 , 1 , 1 , 1 , 0 , 0 , 0 , 0 , 1,  //4
 1 , 0 , 0 , 0 , 1 , 0 , 0 , 0 , 0 , 1,  //5
 1 , 0 , 1 , 0 , 0 , 0 , 1 , 0 , 0 , 1,  //6
  1 , 0 , 1 , 1 , 1 , 0 , 1 , 1 , 0 , 1,  //7
  1 , 1 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 1,  //8
  1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1  //9
} ;

分析:

1、可以利用C语言的栈来实现,起点坐标为(1,1),如果该步能走通,把该步的坐标入栈,且对该坐标进行标记,代表已经走过。

2、然后以该坐标(x,y)为中心进行,对该坐标的上(x,y-1)、右(x+1,y)、下(x,y-1)、左(x-1,y)的坐标顺时针进行判断可否走通(可否走通的条件为:该坐标的值为0,同时该步没有走过),寻找下一步坐标,如果找到了下一步,就接着入栈。

3、如果判断出来该坐标的四周的四个坐标(上、右、下、左)都不能够走通,则把该坐标出栈。

4、如果栈不空则接着往下找,如果栈空,则结束,没有可行的通路。

5、这样一直进行,找到出口坐标则结束,找到可行的通路

下面是我写的代码:有问题大家多交流。

#include
#include

//迷宫的布局放入到一个二维的数组中
      //0 , 1 , 2 , 3 , 4 , 5 , 6 , 7 , 8 , 9
const int mizu[10][10] = {1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1,  //0
        1 , 0 , 0 , 1 , 0 , 0 , 0 , 1 , 0 , 1,  //1
        1 , 0 , 0 , 1 , 0 , 0 , 0 , 1 , 0 , 1,  //2
        1 , 0 , 0 , 0 , 0 , 1 , 1 , 0 , 0 , 1,  //3
        1 , 0 , 1 , 1 , 1 , 0 , 0 , 0 , 0 , 1,  //4
        1 , 0 , 0 , 0 , 1 , 0 , 0 , 0 , 0 , 1,  //5
        1 , 0 , 1 , 0 , 0 , 0 , 1 , 0 , 0 , 1,  //6
          1 , 0 , 1 , 1 , 1 , 0 , 1 , 1 , 0 , 1,  //7
          1 , 1 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 1,  //8
          1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1 , 1  //9
      } ;
int flag[10][10] = {0} ;     
typedef int TypeData ;
typedef struct node{
 TypeData datax ;
 TypeData datay ;
 struct node* next ;
}Node , *LinkList ;

typedef struct stack{
 LinkList top ;
}STACK ;
//************************函数声明******************************
int stackInit(STACK* s ) ;
int pushStack(STACK* s ,TypeData x , TypeData y) ;
void popStack(STACK* s , TypeData* x , TypeData* y) ;
int isStackEmpty(STACK* s) ;
int mizePath(STACK* s ,TypeData end_x , TypeData end_y ) ;
int passMizu(TypeData x , TypeData y ) ;
//**********************************************************

//链栈的初始化
int stackInit(STACK* s )
{
 LinkList p = (LinkList)malloc(sizeof(Node)) ;
 if(!p)  return -1 ;
 p->next = NULL ;
 p->datax = -1 ; 
 p->datay = -1 ;
 s->top = p ;
 return 1 ;
}
//入栈操作
int pushStack(STACK* s ,TypeData x , TypeData y)
{
 LinkList p = (LinkList)malloc(sizeof(Node)) ;
 if(!p)  return -1 ;  //分配内存失败
 p->datax = x ;
 p->datay = y ;
 p->next = s->top ;
 s->top = p ;
 return 1 ;
}
//出栈的操作
void popStack(STACK* s , TypeData* x , TypeData* y)
{
 LinkList p = s->top ;
 s->top = p->next ;
 (*x) = p->datax ;
 (*y) = p->datay ;
 free(p) ;
}
//判断栈是否为空
//1 空  0 不空
int isStackEmpty(STACK* s)
{
 if(s->top->datax == -1)  //栈空
  return 1 ;
 return 0 ; 
}
//找到最佳路径
//end_x , end_y为结束的坐标
//上 、右、下、左寻找方式
int mizePath(STACK* s ,TypeData end_x , TypeData end_y )
{
 pushStack(s , 1 ,1) ; //初始坐标压栈
 TypeData nowx = 1 , nowy = 1 ;
 flag[nowx][nowy] = 1 ;  //该坐标已经被占用不能再通过
 while(!isStackEmpty(s)) //当栈不空的时候
 {
  if((nowx == end_x)&&(nowy == end_y))
  {
   return 1 ;
  }
//  printf(“nowx = %d  nowy = %d\n”,nowx , nowy) ;
//  system(“PAUSE”) ;
  if(passMizu(nowx , nowy-1))  //先向上寻找
  {
   nowy = nowy – 1 ; //坐标更改
   pushStack(s , nowx , nowy) ;  //把该坐标压栈
   flag[nowy][nowx] = 1 ;  //该坐标已经被占用不能再通过
  }
  else  if(passMizu(nowx+1 , nowy)) //向右寻找 
  {
   nowx = nowx + 1 ;
   pushStack(s , nowx , nowy) ;  //把该坐标压栈
   flag[nowy][nowx] = 1 ;  //该坐标已经被占用不能再通过
  }
  else if(passMizu(nowx , nowy+1)) //向下寻找
  {
   nowy = nowy + 1 ;
   pushStack(s , nowx , nowy) ;  //把该坐标压栈
   flag[nowy][nowx] = 1 ;  //该坐标已经被占用不能再通过
  }
  else if(passMizu(nowx-1 , nowy)) //向左寻找
  {
   nowx = nowx – 1 ;
   pushStack(s , nowx , nowy) ;  //把该坐标压栈
   flag[nowy][nowx] = 1 ;  //该坐标已经被占用不能再通过
  }
  else  //都行不通
  {
   popStack(s , &nowx , &nowy) ;
  }
   
 } //while(!isStackEmpty(s)) //当栈不空的时候
 return 0 ;
}
//判断该位置是否可通
int passMizu(TypeData x , TypeData y )
{
 if((x > 9)||(y > 9))
  return 0 ;  //越界不可通
 if(mizu[y][x])
  return 0 ;  //该位置是墙,不可通
 if(flag[y][x]) 
  return 0 ;  //该坐标已经被占用,不能通过
 return 1 ;   
}
//打印栈中的数据
int printStackData(STACK s)
{
 STACK temp ;  //新建一个临时的栈
 stackInit(&temp) ;
 if(s.top->datax == -1)  return 0 ; //栈为空
 while(s.top->datax != -1)
 {
  pushStack(&temp,s.top->datax,s.top->datay);
  s.top = s.top->next ;
 }
 while(temp.top->datax != -1)
 {
  printf(“(%d,%d)\n”,temp.top->datax , temp.top->datay) ;
  temp.top = temp.top->next ;
 }
 return 1 ;
}
int main()
{
 STACK mi_stack ;
 int i = 0 , j = 0 ;
 stackInit(&mi_stack) ;
 
 if(mizePath(&mi_stack , 8 , 8))
 {
  printStackData(mi_stack) ; //把坐标倒叙打印出来
 } 
 else printf(“没有通路!\n”) ;
 
 return  0 ;
}