迷宫问题求最小路径

迷宫问题求最短路径问题,是相对经典的栈应用问题,我们将每一个通路的坐标使其入栈,当它遇到闭路时在让其出栈,当找到一个路径时,将其保存到另一个最小栈中。(用迭代实现)
下面是问题描述:
在一个地图里面,0代表通路,1代表墙壁,从左边进入,其他方向出去,算一条通路。
这里写图片描述

问题解决:
这里写图片描述
最终实现结果:
这里写图片描述
下面是代码实现:

#include<stack>
#include<iostream>
#include<assert.h>
#include <fstream>
using namespace std;

const int N = 10;
struct Pos
{
    size_t _row; //行
    size_t _col; //列

};

void InitMaze(int maze[][N], size_t N)  //在文件中定义迷宫,并写入程序
{
    FILE* fin = fopen("map.txt","r");
    assert(fin);
    for (size_t i = 0; i < N; i++)
    {
        for (size_t j = 0; j < N;)
        {
            int value = getc(fin);
            if (value == '0' || value == '1')
            {
                maze[i][j] = value - '0';
                j++;
            }
            else if (value == EOF)
                cout << "访问出错" << endl;
        }
    }

}
void PrintMaze(int maze[][N],size_t N)   //打印迷宫
{
    for (size_t i = 0; i < N; i++)
    {
        for (size_t j = 0; j < N; j++)
        {
            cout << maze[i][j]<<"  ";
        }
        cout << endl;
    }
    cout << endl;
}

bool CheckAccess(int maze[][N], size_t N, Pos cur,Pos next)  // 判断位置是否合法,并且位置是否为通路
{
    assert(maze);
    if (next._col < 0 || next._col >= N
        || next._row < 0 || next._row >= N
        || maze[next._row][next._col] == 1)
    {
        return false;
    }
    if (maze[next._row][next._col] == 0)
    {
        return true;
    }
    return maze[cur._row][cur._col] < maze[next._row][next._col];
}

void GetMazePathR(int maze[][N], size_t N, Pos cur,stack<Pos>& shortPath, stack<Pos>& path)  //获得最短通路
{
    assert(maze);
    path.push(cur);
    if (cur._row == N - 1 || cur._row == 0
        || cur._col == N-1)
    {
        if (shortPath.empty() || path.size() < shortPath.size())
        {
            shortPath = path;
        }
        path.pop();
        return;
    }
    Pos next;

    //上
    next = cur;
    next._row -= 1;
    if (CheckAccess(maze, N, cur, next))
    {
        maze[next._row][next._col] = maze[cur._row][cur._col] + 1;
        GetMazePathR(maze, N, next, shortPath, path);

    }

    //下
    next = cur;
    next._row += 1;
    if (CheckAccess(maze, N, cur, next))
    {
        maze[next._row][next._col] = maze[cur._row][cur._col] + 1;
        GetMazePathR(maze, N, next, shortPath, path);
    }
    //左
    next = cur;
    next._col -= 1;
    if (CheckAccess(maze, N, cur, next))
    {
        maze[next._row][next._col] = maze[cur._row][cur._col] + 1;
        GetMazePathR(maze, N, next, shortPath, path);
    }
    //右
    next = cur;
    next._col += 1;
    if (CheckAccess(maze, N, cur, next))
    {
        maze[next._row][next._col] = maze[cur._row][cur._col] + 1;
        GetMazePathR(maze, N, next, shortPath, path);
    }
    path.pop();
}



void PtintRose(stack<Pos>& shortPath)  //打印最短通路路径
{
    int size = shortPath.size();
    int row[10];
    int col[10];
    for (int i = 0; i < size;i++)
    {
        row[i] = shortPath.top()._row;
        col[i] = shortPath.top()._col;
        shortPath.pop();
    }
    for (int i = size; i > 0; i--)
    {
        cout << "第" << size-i+1 << "个位置: ";
        cout << "行:" << row[i-1]<<" ";
        cout << "列:" << col[i-1];
        cout << endl;
    }
}
void TestMaze()
{
    int maze[N][N];
    InitMaze(maze, N);
    PrintMaze(maze, N);
    Pos cur;
    cur._row = 1;
    cur._col = 0;
    maze[1][0] = 2;
    stack<Pos> shortPath;
    stack<Pos> path;


    GetMazePathR(maze, N, cur, shortPath, path);
    PrintMaze(maze, N);
    cout << shortPath.size() << endl;
    PtintRose(shortPath);
}
  • 3
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
问题描述: 以一个m*n的长方阵表示迷宫,0和1分别表示迷宫中的通路和障碍。设计一个程序,对任意设定的迷宫,出从入口(0,0)到出口(m-1,n-1)的通路通路总数,或得出没有通路的结论。例如下图, 0(入口) 1 0 1 0 0 0 0 1 0 0 1 1 0 0 0 0 0 0 0 0 1 0 0 0(出口) 从入口到出口有6条不同的通路。 而下图: 0 0 1 0 0 0 1 0 0 0 1 0 0 0 1 0 0 0 0 0 1 1 0 1 0 1 1 1 0 0 1 0 0 0 0 1 0 0 0 0 0 1 0 0 0 1 0 1 0 1 1 1 1 0 0 1 1 1 0 0 0 1 0 1 1 1 0 0 0 0 0 0 从入口到出口则没有通路。 算法设计: 给定一个m*n的长方阵表示迷宫,设计算法输出入口到出口的通路通路总数,或得出没有通路的结论。 算法提示: 和皇后问题与分书问题类似。可以用二维数组存储迷宫数据,对于迷宫中任一位置,均可约定有东、南、西、北四个方向可通。从当前位置a(用(x,y)表示一个位置,假定它是以向右的x轴和向下的y轴组成的平面上的一个点)出发依次尝试四个方向是否有路,若某个方向的位置b可通,则按照同样的方法继续从b出发寻找。若到达出口,则找到一条通路。 数据输入: 由文件input.txt 提供输入数据。第一行是m和n的值,空格分隔,其后共m行。每行有n个数字,数和数之间用空格分隔。 结果输出: 将计算出的所有从入口到出口的通路输出到文件output.txt 中。若没有通路,则将0写入文件中。
#include #include #define N1 9 #define N2 8 #define T N1*N2 #define M 4 char B[N1+1][N2+1]; int count=0; //记录路径条数 typedef struct node1 { int a1; int a2; }find,direct[M+1]; typedef struct { int b1; int b2; int id; }site; typedef struct //顺序 { site ht[T]; int top; }Stack; void Push(Stack *s,int a,int b) { s->top++; s->ht[s->top].b1=a; s->ht[s->top].b2=b; } void Gettop(Stack * s,int *a,int *b) { *a=s->ht[s->top].b1; *b=s->ht[s->top].b2; } void create(char *a) //从文件读出迷宫(正确) { int i=0,j=0,p=1; char x; FILE *fp; fp=fopen("in.txt","r"); if(fp==NULL) { printf("文件不能打开!\n"); exit(0); } x=fgetc(fp); while(x!=EOF) { if(x=='0') { i++; a[i]=x; } if(x=='1') { i++; a[i]=x; } x=fgetc(fp); } printf(" ~~~~~~~生成迷宫~~~~~~~\n"); x=fgetc(fp); while(p<=T) //用二维数组b记录迷宫每个位置是否可行 { for(i=1;i<=N1;i++) for(j=1;j<=N2;j++) { B[i][j]=a[p]; p++; } } printf(" "); printf("■■■■■■■■■■■■\n"); printf(" ■"); printf(" ■\n"); for(i=1;i<=N1;i++) { printf(" "); printf("■ "); for(j=1;jht[s1->top].id=id; B[x][y]='*'; while(s1->top>0) { Gettop(s1,&x,&y); id=s1->ht[s1->top].id; if(x==B1&&y==B2) { count++; fprintf(fp,"%d%c%c",count,':',' '); printf("第 %d 条路径(长度为%d):\n",count,s1->top); s1->ht[s1->top].id=0; for(i=1;itop;i++) { printf("(%d,%d,%d)->",s1->ht[i].b1,s1->ht[i].b2,s1->ht[i].id); fprintf(fp,"%c%d%c%d%c%d%c%c",'(',s1->ht[i].b1,',',s1->ht[i].b2,',',s1->ht[i].id,')',' '); if(i==0) fprintf(fp,"%c%c%c%c",'\n',' ',' ',' '); if(i%8==0) printf("\n"); } fprintf(fp,"%c",'\n'); printf("结束!\n\n"); if(s1->toptop=s1->top; min=s1->top; for(i=1;itop;i++) s2->ht[i]=s1->ht[i]; } B[x][y]='0'; s1->top--; //退(s1->top--) Gettop(s1,&x,&y); id=s1->ht[s1->top].id; } fun=0; while(idht[s1->top].b1; y=s1->ht[s1->top].b2; x=x+p[id].a1; y=y+p[id].a2; if(x==0||y==0||x>N1||y>N2) continue; if(B[x][y]=='0') { fun=1; break; } } if(fun==1) //找到通路 { s1->ht[s1->top].id=id; Push(s1,x,y); B[x][y]='*'; s1->ht[s1->top].id=0; } else { x=s1->ht[s1->top].b1; y=s1->ht[s1->top].b2; B[x][y]='0'; s1->top--; } } if(count==0) printf(" 无路径!\n"); else { printf("\n\n\n "); printf("所有路径已存储在文件%s 中,请去查找!\n\n",filename); } return 1; } void Print(Stack *s2,char filename[]) { int i; FILE *fp; fp=fopen(filename,"a+"); if(fp==NULL) { printf("文件不能打开!\n"); exit(0); } if(count!=0) { fprintf(fp,"%s","最短路径为:"); fprintf(fp,"%c",'\n'); printf(" "); printf("%s\n","**********最短路径**********\n"); for(i=1;itop;i++) { printf("(%d,%d,%d) ->",s2->ht[i].b1,s2->ht[i].b2,s2->ht[i].id); fprintf(fp,"%c%d%c%d%c%d%c%c",'(',s2->ht[i].b1,',',s2->ht[i].b2,',',s2->ht[i].id,')',' '); if(i==0) fprintf(fp,"%c",'\n'); if(i%7==0) printf("\n"); } fprintf(fp,"%c",'\n'); printf("结束!\n"); printf("(最短路径长度: %d)\n",s2->top); } } void main() //主函数 { char a[T+1]; //二维数组b记录迷宫的每个位置 char filename1[20],filename2[20]; int x1,x2,y1,y2,k; Stack *s1,*s2; direct f1; f1[1].a1=0; f1[1].a2=1; //判断方向(右) f1[2].a1=1; f1[2].a2=0; //(下) f1[3].a1=0; f1[3].a2=-1; //(左) f1[4].a1=-1; f1[4].a2=0; //(上) s1=(Stack *)malloc(sizeof(Stack)); s2=(Stack *)malloc(sizeof(Stack)); s1->top=0; //指向顶(初始化) s2->top=0; create(a); printf("\n\n "); printf("请输入入口坐标: "); scanf("%d%d",&x1,&x2); printf(" "); printf("请输入出口坐标: "); scanf("%d%d",&y1,&y2); printf(" "); printf("请输入存储所有路径的文件名:"); scanf("%s",filename1); printf(" "); printf("请输入存储最短路径的文件名:"); scanf("%s",filename2); system("cls"); k=search(x1,x2,y1,y2,s1,s2,f1,filename1); if(k==1) Print(s2,filename2); printf("\n"); }
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值