写给妹妹的编程札记 5 - 搜索: 迷宫问题 - 广度优先搜索

2024-05-24 04:58

本文主要是介绍写给妹妹的编程札记 5 - 搜索: 迷宫问题 - 广度优先搜索,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

        让我们也使用广度优先搜索来解决一下迷宫问题,可以对比一下《写给妹妹的编程札记 4 - 搜索: 迷宫问题 - 深度优先搜索》。

        如在《写给妹妹的编程札记 3 - 穷举: 深度优先搜索/广度优先搜索》中描述的广度优先搜索, 对一个简单的例子,我们手动进行一遍迷宫遍历。每次找到队首的搜索状态,把从这个状态开始的全部状态加入队列。



        广度优先搜索, 我们需要一个队列来维护访问过的搜索状态。 对于每个搜索状态,我们记录该状态对应迷宫中的位置currX/currY, 从上一个状态怎么过来的: prevDir, 上一个状态在队列中的位置prevState。 后面两个信息prevDir/prevState主要是为了反向找到路径。

typedef struct tagMazeState {int currX;int currY;int prevDir;int prevState;
} MazeState;


        除了新创建的队列,跟深度优先搜索一样, 我们需要记录一个位置是否访问过 - isVisited数组。

void Search(char** maze, int row, int col)
{    bool** isVisited;isVisited = new bool*[row];for(int i = 0; i < row; i++) {isVisited[i] = new bool[col];memset(isVisited[i], 0, sizeof(bool) * col);}MazeState* queue = new MazeState[row * col];int queueHead = 0;int queueTail = 0;MazeState origin;origin.currX     = 0;origin.currY     = 0;origin.prevDir   = -1;origin.prevState = -1;// 判断起点是否为空位if (maze[0][0] != '.') {printf("Invalid maze. Origin maze[0][0] is blocked!\n");return;}// 判断是否已经到达终点if (origin.currX == row - 1 && origin.currY == col - 1) {printf("Already arrive target position!\n");return;}queue[queueTail++] = origin;isVisited[0][0] = 1;BFS(maze, row, col, isVisited, queue, queueHead, queueTail);for(int i = 0; i < row; i++) delete[] isVisited[i];delete[] isVisited;delete[] queue;
}


        有了搜索状态队列的支持, 广度优先搜索的代码就很简单了:

void BFS(char** maze, int row, int col, bool** isVisited, MazeState* queue, int queueHead, int queueTail)
{// 判断队列是否为空while (queueHead < queueTail) {// 取出队首的搜索状态int currX = queue[queueHead].currX;int currY = queue[queueHead].currY;printf("queueHead: (%d, %d)\n", currX, currY);// 检查从这个状态开始的所有能够到达的状态 (广度优先搜索)for (int k = 0; k < 4; k++) {int nextX = currX + direction[k][0];int nextY = currY + direction[k][1];// 检查保证新位置在迷宫内,没有离开迷宫, 且是空位(不能走到墙里面)if ((nextX >= 0) && (nextX < row) && (nextY >= 0) && (nextY < col) && (maze[nextX][nextY] == '.')) {// 找到可以走的方向后, 需要判断这个新的位置是否已经走过,如果已经走过,我们继续会走就出现环路if (!isVisited[nextX][nextY]) {// 标记搜索访问过isVisited[nextX][nextY] = 1;// 没有走过的新位置,加入队列 (广度优先搜索)MazeState nextState;nextState.currX     = nextX;nextState.currY     = nextY;nextState.prevDir   = k;nextState.prevState = queueHead;queue[queueTail++]  = nextState;// 一旦发现目标状态到达的话, 说明已经找到最短路径,输出if (nextX == row - 1 && nextY == col - 1) {// 输出最短路径DisplayPath(queue, queueTail - 1);// 终止搜索return;}}}}// 队首的搜索状态已经完成,从队列中删除 (没有实际删除,还在queue数组中,只是不在逻辑队列中[queueHead, queueTail) )queueHead++;}
}


        跟深度优先搜索不同的是,我们采取广度优先,多条路径同时进行,不能只使用一个path数组保存当前的路径。 对应于搜索状态队列中的每一个状态, 其实都对应着一条从起点开始到该状态的一条路径。为了反向找到这条路径,搜索状态队列中的每个状态都已经记录了从哪个状态过来的,怎么过来的。 我们需要反向找到这条路径:


// 输出从起点开始到(queue[targetPositionIndex].currX, queue[targetPositionIndex].currY)的路径
void DisplayPath(MazeState* queue, int targetPositionIndex)
{// 终止条件。 起点的前一个状态index为:-1if (targetPositionIndex < 0) {return;}// 前一个状态int previousPositionIndex = queue[targetPositionIndex].prevState;// 从前一个状态怎么过来的int previousDirection     = queue[targetPositionIndex].prevDir;// 先输出从起点到前一个状态的路径DisplayPath(queue, previousPositionIndex);// 再输出从前一个状态到当前状态的路径switch(previousDirection) {case 0: printf("N"); break;case 1: printf("S"); break;case 2: printf("W"); break;case 3: printf("E"); break;}
}




这篇关于写给妹妹的编程札记 5 - 搜索: 迷宫问题 - 广度优先搜索的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/997439

相关文章

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

认识、理解、分类——acm之搜索

普通搜索方法有两种:1、广度优先搜索;2、深度优先搜索; 更多搜索方法: 3、双向广度优先搜索; 4、启发式搜索(包括A*算法等); 搜索通常会用到的知识点:状态压缩(位压缩,利用hash思想压缩)。

hdu1240、hdu1253(三维搜索题)

1、从后往前输入,(x,y,z); 2、从下往上输入,(y , z, x); 3、从左往右输入,(z,x,y); hdu1240代码如下: #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#inc

hdu1180(广搜+优先队列)

此题要求最少到达目标点T的最短时间,所以我选择了广度优先搜索,并且要用到优先队列。 另外此题注意点较多,比如说可以在某个点停留,我wa了好多两次,就是因为忽略了这一点,然后参考了大神的思想,然后经过反复修改才AC的 这是我的代码 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<

Linux 网络编程 --- 应用层

一、自定义协议和序列化反序列化 代码: 序列化反序列化实现网络版本计算器 二、HTTP协议 1、谈两个简单的预备知识 https://www.baidu.com/ --- 域名 --- 域名解析 --- IP地址 http的端口号为80端口,https的端口号为443 url为统一资源定位符。CSDNhttps://mp.csdn.net/mp_blog/creation/editor

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal

购买磨轮平衡机时应该注意什么问题和技巧

在购买磨轮平衡机时,您应该注意以下几个关键点: 平衡精度 平衡精度是衡量平衡机性能的核心指标,直接影响到不平衡量的检测与校准的准确性,从而决定磨轮的振动和噪声水平。高精度的平衡机能显著减少振动和噪声,提高磨削加工的精度。 转速范围 宽广的转速范围意味着平衡机能够处理更多种类的磨轮,适应不同的工作条件和规格要求。 振动监测能力 振动监测能力是评估平衡机性能的重要因素。通过传感器实时监

缓存雪崩问题

缓存雪崩是缓存中大量key失效后当高并发到来时导致大量请求到数据库,瞬间耗尽数据库资源,导致数据库无法使用。 解决方案: 1、使用锁进行控制 2、对同一类型信息的key设置不同的过期时间 3、缓存预热 1. 什么是缓存雪崩 缓存雪崩是指在短时间内,大量缓存数据同时失效,导致所有请求直接涌向数据库,瞬间增加数据库的负载压力,可能导致数据库性能下降甚至崩溃。这种情况往往发生在缓存中大量 k

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)