0

0

c++中如何实现深度优先搜索_c++ DFS算法图论遍历方法【详解】

冰火之心

冰火之心

发布时间:2026-01-18 14:26:02

|

832人浏览过

|

来源于php中文网

原创

DFS递归必须用visited标记防环,且需在递归前设true;迭代版用stack并倒序压入保序;判断环需区分无向图(传parent)和有向图(用recStack);邻接表优选vector以提升缓存性能。

c++中如何实现深度优先搜索_c++ dfs算法图论遍历方法【详解】

DFS 递归实现必须处理访问标记

不加访问标记的 DFS 在图中会无限循环,尤其遇到环或无向图边双向存在时。用 vectorunordered_set 记录已访问节点是硬性要求。

  • visited[i] = true 必须在进入递归前设置,不能放在递归调用之后
  • 对邻接表 vector> graph,遍历 graph[u] 中每个 v 前先检查 !visited[v]
  • 无向图中,即使边只存一次(如 u→v),v→u 仍可能通过其他路径抵达,所以标记不可省

迭代版 DFS 要用 stack 模拟调用

手动维护栈比递归更可控,适合防止爆栈或需记录路径/层数的场景。注意:栈中只存节点编号,不存边或状态标志。

  • 入栈顺序影响遍历结果——若按邻接表原始顺序压栈,出栈是逆序;想保持左→右顺序,应倒序压入
  • 每次 stack.pop() 后立即设 visited[u] = true,避免重复入栈
  • 不要在循环内对当前节点的邻居反复 push/pop,应一次性遍历并过滤已访问节点
void dfs_iterative(int start, const vector>& graph) {
    vector visited(graph.size(), false);
    stack stk;
    stk.push(start);

    while (!stk.empty()) {
        int u = stk.top();
        stk.pop();
        if (visited[u]) continue;
        visited[u] = true;
        // 处理节点 u:打印、存路径、更新状态等

        // 倒序压入,保证邻接表顺序被保留(可选)
        for (auto it = graph[u].rbegin(); it != graph[u].rend(); ++it) {
            if (!visited[*it]) {
                stk.push(*it);
            }
        }
    }
}

DFS 判断连通性与找环的逻辑差异

同一套 DFS 框架,仅靠参数和返回值设计就能区分用途。关键在如何定义“回边”和是否允许父节点干扰判断。

LALALAND
LALALAND

AI驱动的时尚服装设计平台

下载
  • 判断无向图是否有环:传入 parent 参数,跳过 v == parent 的边;发现未访问却已入栈的邻居即为环
  • 判断有向图环:需额外 recStack 数组记录当前递归路径上的节点,而非仅 visited
  • 连通分量计数:每启动一次未访问节点的 DFS,components++,适用于非连通图

vector> 邻接表比 map> 更快但不支持稀疏大编号

下标连续且范围已知(如节点编号 0~n−1)时,vector 是首选;若节点编号稀疏(如 1e9 级别 ID),必须用哈希结构,但 DFS 本身开销会上升。

立即学习C++免费学习笔记(深入)”;

  • vector> graph(n) 支持 O(1) 随机访问,push_back 平摊 O(1)
  • mapunordered_map 时,每次 graph[u].count(v) 是 O(log k) 或均摊 O(1),但整体常数更大
  • DFS 中反复调用 graph[u].size() 和遍历,vector 的 cache 局部性明显更好
真正容易被忽略的是:递归 DFS 的函数参数传递方式(值传 vs 引用)、迭代 DFS 中栈元素是否去重、以及有向图环检测时 recStack 的置位/复位时机——这些细节不写错,算法才能稳定输出正确结果。

相关专题

更多
counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

197

2023.11.20

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

318

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

538

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

52

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

391

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

572

2023.08.10

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

75

2025.09.05

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

43

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号