C语言DFS和BFS解决迷宫问题

2023-05-16

C语言DFS与BFS

迷宫问题

题目描述

给定一个 N \times MN×M 方格的迷宫,迷宫里有 TT 处障碍,障碍处不可通过。

在迷宫中移动有上下左右四种方式,每次只能移动一个方格。数据保证起点上没有障碍。

给定起点坐标和终点坐标,每个方格最多经过一次,问有多少种从起点坐标到终点坐标的方案。

输入格式

第一行为三个正整数 N,M,TN,M,T,分别表示迷宫的长宽和障碍总数。

第二行为四个正整数 SX,SY,FX,FYSX,SY,FX,FY,SX,SYSX,SY 代表起点坐标,FX,FYFX,FY 代表终点坐标。

接下来 TT 行,每行两个正整数,表示障碍点的坐标。

输出格式

输出从起点坐标到终点坐标的方案总数。

解题思路

这个问题,方案总数的话还是要用DFS(深度搜索优先)。深搜是一条路走到黑,不撞南墙不回头。需要一个二维数组记下走过的或者不能走的(这里指障碍),我们把这些情况在book数组里面统统记作1(没走用0表示),然后就是每一次递归我们都确定一个坐标。如何确定呢,就是每一个都试一试(上下左右),看满不满足条件(条件是不为1)。满足然后往下递归,记得book数组对应记1。这里有一个很重要的点(这里采用的是回溯算法),递归回来的时候我们要把这个点当作它没有出现继续试试另外的路(就是把book记为0)。结束递归,就是判断当前坐标是否到了终点坐标,用全局变量记数(当然你也可以把它加入函数参数来每次递归计数)。

最短路径这个问题就得看BFS(广度搜素优先)。这个是每一个坐标都试一次,把所有出现的情况都记下来。这个算法牵扯到队列,这也是利用队列先进先出的特点来做。我们定义一个结构体里面包含x,y的坐标,还有s(从开始到这要走的步数),这个结构体我们定义M*N+1大小的队列(为什么是+1因为这是要判断结束的条件,而且最多就是M*N个,因为每一个只找一次),从head开始记录起始坐标,tail则到head后面去,用来保留要存进去的坐标。我们同样用book数组来记录是否出现或者是障碍。我们从当前坐标能走开始(上下左右),按照你想要的顺序来。然后看它是否满足条件(遇到障碍或者走过也就是1的情况),满足记录下来,我们要把这个坐标所有能延生的点都记录下来,tail记录之后++,记得s也要在head的s的基础上+1(从head延生的,当然要+1,走了一步了),book要记下来走过了,这跟深搜不一样,广搜每一个点之走一次,所以没关系。这个坐标举完了之后要记得head++,head就到下一个能够延生的坐标了,在在这个点继续延生。在其中我们判断是否到达了终点坐标,第一次到的一定是最短路径,这个因为,是最短才会最快出现。这个时候记录下来当前tail的s的值,break即可。

#include<stdio.h>
#include<malloc.h>
#define N 5
#define M 5
int book[M][N]={0},count=0;
int start[2]={0},end[2]={0};
int dfs(int x,int y)
{
	int i,j,tx,ty;
	if(x==end[0]&&y==end[1])
	{
		count++;
		return 0;
	}
	
	int k[4][2]={{-1,0},{0,1},{1,0},{0,-1}};
	for(i=0;i<4;i++)
	{
		tx=x+k[i][0];ty=y+k[i][1];
		if(tx>=1&&tx<=m&&ty>=1&&ty<=n)
			if(book[tx][ty]==0)
			{
				book[tx][ty]=1;
				dfs(tx,ty);
				book[tx][ty]=0;
			}
	}
}

typedef struct node
{
    int x;
    int y;
    int s;
    
}NODE; 
NODE que[M*N+1];
int head=1,tail=2,min=M*N;
int bfs()
{
    int k[4][2]={{-1,0},{0,1},{1,0},{0,-1}},flag=0;
    int i,j,tx,ty;
    while(head<tail)
    {
        for(i=0;i<4;i++)
        {
            tx=que[head].x+k[i][0];
            ty=que[head].y+k[i][1];
            if(tx>=start[0]&&tx<=end[0]&&ty>=start[1]&&ty<=end[1])
                if(book[tx][ty]==0)
                {
                    que[tail].x=tx;
                    que[tail].y=ty;
                    que[tail].s=que[head].s+1;
                    if(tx==end[0]&&ty==end[1])
                        if(min>que[tail].s) 
                        {
                            min=que[tail].s;
                            flag=1;
                            break;
                        }
                    tail++;
                    book[tx][ty]=1;
                }
            
        }
        if(flag==1) break;
        head++;
    }
}
int main() 
{
    int m,n,t,i,j,x,y;//第一个输入的是长,第二个是宽
    scanf("%d%d%d",&n,&m,&t);//我习惯用m表示行,n表示列
    scanf("%d%d%d%d",&start[0],&start[1],&end[0],&end[1]);
    for(i=0;i<t;i++)
    {
        scanf("%d%d",&x,&y);
        book[x][y]=1;
    }    
    dfs(start[0],start[1]);
    que[1].x=start[0];
    que[1].y=start[1];
    que[1].s=0;    
    bfs();
    printf("the min is%d\n",min);
    printf("the total is%d\n",count);
    return 0;
}

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

C语言DFS和BFS解决迷宫问题 的相关文章

  • Maximum Diameter Graph 【CodeForces - 1082D】【搜索+构造】

    题目链接 一开始忘记输出有多少条边 WA了好几发都跑不过第一组测试样例 开始怀疑自己是不是读了道假题 然后在大佬们的帮助下 终于AC 好伤心 读假样例 一定是我太弱了 我的思想是采用了树链剖分的dfs 构造思想 可能是因为最近少用了树链剖分
  • 学渣带你刷Leetcode0017. 电话号码的字母组合

    题目描述 给定一个仅包含数字 2 9 的字符串 返回所有它能表示的字母组合 给出数字到字母的映射如下 与电话按键相同 注意 1 不对应任何字母 示例 输入 23 输出 ad ae af bd be bf cd ce cf 说明 尽管上面的答
  • 搜索学习心得

    在学习了众多搜索的方式后 不由感慨 啊 太巨了 今天huayucaiji我就给大家讲一讲C 搜索的心得吧 深度优先搜索 广度优先搜索 迭代加深搜索 一个一个讲吧 深度优先搜索 深度优先搜索 下简称 深搜 简称DFS 是简洁明了的搜索方式 以
  • 基础算法题——德邦国王(dfs、剪枝)

    德邦国王 题目还算中规中矩 就是剪枝比较麻烦 解题思路 dfs 剪枝 移动次数不超过设定值 M 若有解 则后面的步骤不可大于该解的值 不断查询完美矩阵与当前矩阵不同的个数 t t 1 为最快可将当前矩阵移动成完美矩阵的步数 若 t 1 已经
  • [LeetCode] Binary Tree Level Order Traversal 二叉树层次遍历(DFS

    目录 1 Binary Tree Level Order Traversal 二叉树层次遍历 BFS 2 Binary Tree Level Order Traversal II 二叉树层次遍历从低往高输出 BFS 3 Maximum De
  • 【2019年ICPC南昌网络赛】Distance on the tree【DFS+线段树合并(可持久化线段树)】

    题目链接 DSM Data Structure Master once learned about tree when he was preparing for NOIP National Olympiad in Informatics i
  • 层序遍历与BFS广度(宽度)遍历搜索算法(C++)

    算法竞赛 file author jUicE g2R qq 3406291309 彬 bin 必应 一个某双流一大学通信与信息专业大二在读 brief 一直在算法竞赛学习的路上 copyright 2023 8 COPYRIGHT 原创技术
  • 图论 笔记

    关于存图 如果是有权值的边 可以用pair define pii pair
  • 【算法】蓝桥杯dfs深度优先搜索之凑算式总结

    本文 算法 蓝桥杯dfs深度优先搜索之凑算式总结 相关文章 算法 蓝桥杯dfs深度优先搜索之排列组合总结 算法 蓝桥杯dfs深度优先搜索之图连通总结 前言 曾几何时这个词现在用正适合不过了 曾几何时我还是对dfs算法一脸懵x的状态 虽说大二
  • BZOJ4345 [POI2016]Korale

    在病房里日题真是一种独特的体验 首先考虑求第一问 我们先把所有元素排序 我们用优先队列维护选数的集合 对每个集合维护集合里的元素的和v和最后一个元素 即最大的元素 lst 初始的时候我们把只包含最小元素的集合推入队列 那么我们取出一个队头元
  • CMake:递归检查并拷贝所有需要的DLL文件

    文章目录 1 目的 2 设计 整体思路 多层依赖的处理 获取 DLL 所在目录 探测剩余的 DLL 文件 3 代码实现 判断 stack 是否为空 判断 stack 是否为空 获取所有 target 检测并拷贝 DLL 4 使用 1 目的
  • 岛屿类-网格类问题-DFS

    本文讲解200 岛屿数量问题 属于常见的岛屿类 网格类问题 本题使用DFS的思想 1 题目 给你一个由 1 陆地 和 0 水 组成的的二维网格 请你计算网格中岛屿的数量 岛屿总是被水包围 并且每座岛屿只能由水平方向和 或竖直方向上相邻的陆地
  • PowerOJ2512: 小红灌溉【染色】

    题目链接 划重点 每个有菜的点只能浇一次且恰好一次 所以意思就是 譬如某个菜的位置是 x y 那么 行x 列y的浇水方案只能使用其中的一个 以此类推 我们给每个有蔬菜的位置的 x y 的x点与y点链接一条无向边 代表x和y只能选择其中的一个
  • Codeforces-1454E Number of Simple Paths(基环树-思维)

    题目大意 给你n个点 n条边 求图中简单路径的个数 题目思路 n个点n条边 那么图中一定有一个环 拿这个图来讲 我们将两点间的关系分为4种 1 两点都在环上 简单路径的个数为2 例如2与5 2 一个点在环上一个点不在环上 简单路径个数为2
  • 不同岛屿的数量

    694 不同岛屿的数量 这道题要给出不同岛屿的数量 最直观的想法就是对发现的不同岛屿进行序列化 然后把序列化结果存到HashSet 那怎么序列化呢 其实比较类似于二叉树的序列化 记录遍历顺序 方向 这里用 1 2 3 4 代表上下左右 1
  • Knight Moves_dfs_2018_3_10

    A friend of you is doing research on the Traveling Knight Problem TKP where you are to find the shortest closed tour of
  • 链式前向星存树图和遍历它的两种方法【dfs、bfs】

    目录 一 链式前向星存图 二 两种遍历方法 一 链式前向星存图 n个点 n 1条边 链式前向星把上面的树图存下来 输入 9 代表要存进去n个点 1 2 下面是n 1条边 每条边连接两个点 1 3 1 7 2 4 4 5 4 6 3 8 3
  • 方板围棋吃子换001

    1 描述 130给定一个二维的矩阵 包含 X 和 O 字母 O 找到所有被 X 围绕的区域 并将这些区域里所有的 O 用 X 填充 示例 X X X X X O O X X X O X X O X X 运行你的函数后 矩阵变为 X X X
  • 数据结构——图的DFS(深度优先遍历)- C语言代码实现

    图的深度优先遍历的基本思想 从图中某顶点v出发 1 访问顶点v 2 依次从v的未被访问的邻接点出发 对图进行深度优先遍历 直至图中和v有路径相通的顶点都被访问 3 若此时图中尚有顶点未被访问 则从一个未被访问的顶点出发 重新进行深度优先遍历
  • 无向图染色问题-dfs剪枝

    无向图染色问题 问题描述 给定一个无向图 要求用最少的颜色将节点染色 限制是不能让相邻节点染上相同的颜色 算法 使用dfs 为节点分配不同的颜色进行尝试 计算每种分配所需的颜色数 最终进行回溯 取得最小的颜色数 代码 C include

随机推荐