无向图的Hierholzer算法流程(二)

精妙的深度优先搜索

虽然 Hierholzer 算法的流程看起来较为复杂,但我们可以通过一段精妙的代码,将算法流程中的两个步骤合并到同一个深度优先搜索过程中。

在图论的代码实现中,一条无向边一般会被拆成两条有向边。由于 Hierholzer 算法的流程需要我们删除一条无向边,因此我们需要同时删除一条有向边和它的反向边。这样的操作只有利用链式前向星^6(亦称边表)存储无向图时才能在 的时间复杂度内完成,因此下文的代码实现中,均使用链式前向星存储无向图。

Hierholzer 算法的一个 C++ 实现如下。请注意,为了便于分析与讲解,下面给出的代码实现并不是欧拉回路的最优时间复杂度实现。下文中,我们将提出该实现的一个优化。

复制代码
// 为了方便存储一条边的众多信息,我们定义一个结构体
//
// struct Edge {
//     int fn, nxt;
//     bool del;
// };
//
// 其中:
//   * fn 表示这条有向边的终点
//   * nxt 表示链式前向星中,该有向边的后续边的编号,如果没有后续边则值为 0
//   * del 表示这条边是否被遍历过了
//
// e[] 是一个类型为 Edge 的数组,e[i] 表示链式前向星中编号为 i 的边
// p[] 是一个类型为 int 的数组,p[i] 表示以 i 为起点的第一条边在链式前向星中的编号
// 如果没有以 i 为起点的有向边则 p[i] 的值为 0
//
// 为了方便我们查找某一条有向边的反向边,当我们向链式前向星中加入第 i(i 从 1 到 m)条无向边 x - y 时,
// 我们会将有向边 x -> y 存储在 e[i * 2],有向边 y -> x 存储在 e[i * 2 + 1],
// 这样,e[i] 和 e[i ^ 1] 就互为反向边
//
// ans 是一个类型为 vector<Edge> 的变量,用来存储我们找到的回路

void dfs(int sn) {
    for (int i = p[sn]; i != 0; i = e[i].nxt) {
        // 这条边已经被遍历过了,跳过
        if (e[i].del) continue;
        // 删除有向边 e[i] 和它的反向边 e[i ^ 1]
        e[i].del = e[i ^ 1].del = true;
        // 继续遍历相邻点
        dfs(e[i].fn);
        // 将边 e[i] 加入结果序列中
        ans.push_back(e[i]);
    }
}

调用 dfs(S) 后,ans 里就保存了从节点 S 出发,并回到节点 S 的欧拉回路。需要注意的是,此时 ans 中保存的欧拉回路,是我们求出的欧拉回路的倒序。因此,我们还需要调用 reverse(ans.begin(), ans.end())ans 中所有元素顺序倒转后,才能得到我们想要的欧拉回路。

相关推荐
gihigo19982 小时前
基于蒙特卡洛的异常值剔除(RANSAC + MC置信区间)—MATLAB实现
开发语言·算法·matlab
Asize2 小时前
数组数据结构底层:从灵活到陷阱
前端·javascript·算法
hairenwangmiao2 小时前
B4041 [GESP202409 四级] 区间排序
算法·排序
人道领域3 小时前
【LeetCode刷题日记】47.全排列Ⅱ
java·开发语言·算法·leetcode
漂流瓶jz3 小时前
UVA-1606 两亲性分子 题解答案代码 算法竞赛入门经典第二版
数据结构·算法·向量·aoapc·算法竞赛入门经典·atan2·浮点
Navigator_Z3 小时前
LeetCode //C - 1095. Find in Mountain Array
c语言·算法·leetcode
不会就选b3 小时前
算法日常・每日刷题--<二分查找>1
算法
「維他檸檬茶」3 小时前
大模型算法学习2026.6.13
学习·算法
叫我:松哥3 小时前
基于Python的共享单车租赁数据分析与预测系统,技术栈flask+boostrap+随机森林+XGBoost
人工智能·python·深度学习·算法·随机森林·数据分析·flask