当前位置: 首页 > news >正文

随机深度优先搜索(Randomized DFS)算法原理

随机深度优先搜索是深度优先搜索的变种,通过在每一步随机选择邻接节点来增加路径的不可预测性。该算法天然适合生成或解决迷宫问题,因其倾向于生成长而曲折的路径。

核心特点:

  • 使用栈(显式或隐式)实现回溯
  • 随机选择邻接节点顺序
  • 时间复杂度为O(V+E),空间复杂度为O(V)

C++实现关键步骤

数据结构定义
#include <iostream> #include <vector> #include <stack> #include <cstdlib> #include <ctime> #include <algorithm> struct Cell { int x, y; bool visited = false; bool top_wall = true, bottom_wall = true; bool left_wall = true, right_wall = true; };
迷宫初始化
void initMaze(std::vector<std::vector<Cell>>& maze, int width, int height) { maze.resize(height); for (int y = 0; y < height; ++y) { maze[y].resize(width); for (int x = 0; x < width; ++x) { maze[y][x].x = x; maze[y][x].y = y; } } std::srand(std::time(nullptr)); }
随机DFS核心实现
void generateMaze(std::vector<std::vector<Cell>>& maze) { std::stack<Cell*> cellStack; Cell* current = &maze[0][0]; current->visited = true; cellStack.push(current); while (!cellStack.empty()) { current = cellStack.top(); auto neighbors = getUnvisitedNeighbors(current, maze); if (!neighbors.empty()) { Cell* next = neighbors[std::rand() % neighbors.size()]; removeWalls(current, next); next->visited = true; cellStack.push(next); } else { cellStack.pop(); } } }
辅助函数实现
std::vector<Cell*> getUnvisitedNeighbors(Cell* cell, const std::vector<std::vector<Cell>>& maze) { std::vector<Cell*> neighbors; int dx[] = {0, 1, 0, -1}; int dy[] = {-1, 0, 1, 0}; for (int i = 0; i < 4; ++i) { int nx = cell->x + dx[i]; int ny = cell->y + dy[i]; if (nx >= 0 && nx < maze[0].size() && ny >= 0 && ny < maze.size() && !maze[ny][nx].visited) { neighbors.push_back(const_cast<Cell*>(&maze[ny][nx])); } } return neighbors; } void removeWalls(Cell* a, Cell* b) { if (a->x == b->x) { if (a->y < b->y) { a->bottom_wall = false; b->top_wall = false; } else { a->top_wall = false; b->bottom_wall = false; } } else { if (a->x < b->x) { a->right_wall = false; b->left_wall = false; } else { a->left_wall = false; b->right_wall = false; } } }

迷宫求解实现

路径搜索算法
bool solveMaze(std::vector<std::vector<Cell>>& maze, Cell* start, Cell* end, std::vector<Cell*>& path) { std::stack<Cell*> stack; stack.push(start); start->visited = true; while (!stack.empty()) { Cell* current = stack.top(); path.push_back(current); if (current == end) return true; auto neighbors = getAccessibleNeighbors(current, maze); if (!neighbors.empty()) { Cell* next = neighbors[std::rand() % neighbors.size()]; next->visited = true; stack.push(next); } else { path.pop_back(); stack.pop(); } } return false; }
可通行邻居检测
std::vector<Cell*> getAccessibleNeighbors(Cell* cell, const std::vector<std::vector<Cell>>& maze) { std::vector<Cell*> neighbors; int dx[] = {0, 1, 0, -1}; int dy[] = {-1, 0, 1, 0}; bool walls[] = {cell->top_wall, cell->right_wall, cell->bottom_wall, cell->left_wall}; for (int i = 0; i < 4; ++i) { if (!walls[i]) { int nx = cell->x + dx[i]; int ny = cell->y + dy[i]; if (nx >= 0 && nx < maze[0].size() && ny >= 0 && ny < maze.size() && !maze[ny][nx].visited) { neighbors.push_back(const_cast<Cell*>(&maze[ny][nx])); } } } return neighbors; }

可视化输出

控制台打印迷宫
void printMaze(const std::vector<std::vector<Cell>>& maze) { for (const auto& row : maze) { // 打印顶部墙壁 for (const auto& cell : row) { std::cout << (cell.top_wall ? "+---" : "+ "); } std::cout << "+\n"; // 打印侧边墙壁 for (const auto& cell : row) { std::cout << (cell.left_wall ? "|" : " "); std::cout << " "; } std::cout << "|\n"; } // 打印底部边界 for (size_t i = 0; i < maze[0].size(); ++i) { std::cout << "+---"; } std::cout << "+\n"; }

完整示例调用

int main() { const int WIDTH = 10, HEIGHT = 10; std::vector<std::vector<Cell>> maze; initMaze(maze, WIDTH, HEIGHT); generateMaze(maze); std::vector<Cell*> path; solveMaze(maze, &maze[0][0], &maze[HEIGHT-1][WIDTH-1], path); printMaze(maze); return 0; }

算法优化方向

  • 记忆化搜索:记录已探索路径避免重复计算
  • 双向搜索:从起点和终点同时进行搜索
  • 启发式引导:结合曼哈顿距离等启发式信息
  • 并行化处理:利用多线程加速搜索过程

该实现完整展示了随机DFS在迷宫生成与求解中的应用,通过随机选择邻接节点使得每次生成的迷宫都具有独特性,同时保持算法的高效性。

http://www.jsqmd.com/news/102157/

相关文章:

  • LobeChat插件开发入门:如何为AI聊天界面扩展新功能?
  • 洛谷 P1892 [BalticOI 2003] 团伙 简单并查集 做法 题解
  • Realm端口转发教程
  • 商业模式画布填充:LobeChat理清商业逻辑
  • LobeChat制造业知识库查询终端部署案例
  • 电路中mos管的作用
  • 计算机Java毕设实战-基于javaWEB的餐厅后勤管理系统的设计与实现基于javaWEB的饭馆餐厅后勤管理系统的设计与实现【完整源码+LW+部署说明+演示视频,全bao一条龙等】
  • 计算机Java毕设实战-基于Java的仓库管理系统设计与实现基于Java+SpringBoot+Vue仓库管理系统的设计与实现【完整源码+LW+部署说明+演示视频,全bao一条龙等】
  • 计算机Java毕设实战-基于Java+SpringBoot+Vue的畅销图书推荐系统基于java的畅销图书推荐系统【完整源码+LW+部署说明+演示视频,全bao一条龙等】
  • 计算机Java毕设实战-基于JavaWeb的心聘求职平台的设计与实现基于springboot的人才求职招聘平台设计与实现【完整源码+LW+部署说明+演示视频,全bao一条龙等】
  • 计算机Java毕设实战-基于JavaWeb的家装一体化平台室内设计、装修施工、建材选购【完整源码+LW+部署说明+演示视频,全bao一条龙等】
  • 计算机Java毕设实战-基于javaweb的宠物托管系统基于springboot+vue的宠物托管系统【完整源码+LW+部署说明+演示视频,全bao一条龙等】
  • 计算机Java毕设实战-基于javaweb的在线图书借阅管理系统图书馆在线借阅管理系统【完整源码+LW+部署说明+演示视频,全bao一条龙等】
  • 计算机Java毕设实战-基于JavaWeb的兽医站管理系统的设计与实现动物医院管理系统的设计与实现【完整源码+LW+部署说明+演示视频,全bao一条龙等】
  • 大数据领域数据增强:挖掘数据潜力的秘诀
  • Java 并发编程之 ThreadPoolExecutor 深度解析:从原理到实战
  • 智能体间博弈理论在价值投资策略优化中的应用
  • 巴菲特的投资观察与启示
  • 芒格的“反向工程“思维在量子密码破解防御中的应用
  • Python面向对象——进阶(三)
  • CosyVoice3 - 跨语言、会方言、懂情绪的智能配音工具 文本转语音 语音克隆 支持50系显卡 一键整合包下载
  • 四季梅豆角矮砧密植:水肥一体化系统的铺设要点
  • 【LLM基础教程】从序列切分到上下文窗口02_三种数据切分方法
  • 9 个高效降AI率工具,自考人必备!
  • 【LLM基础教程】LLM训练数据集是如何构造的:从文档到Token Block
  • 8 个降AI率工具推荐,本科生高效避坑指南
  • 打表小技巧
  • 10个降AI率工具,专科生高效避坑指南
  • 8 个降AI率工具,自考学生高效避坑指南
  • SQL中的NULL值处理技巧