0

0

c++中如何实现克鲁斯卡尔算法_c++ Kruskal算法代码实现

尼克

尼克

发布时间:2026-01-04 13:06:51

|

484人浏览过

|

来源于php中文网

原创

Kruskal易错因排序失效和并查集未优化:边须权重前置或自定义比较,UF必须路径压缩,验证需检查选边数是否为n-1且图连通。

c++中如何实现克鲁斯卡尔算法_c++ kruskal算法代码实现

为什么直接写 Kruskal 常常连边都排不对?

因为 std::sort 默认按第一个字段升序,而你传的边结构体或 std::tuple 如果没重载比较或写错 compare,排序就失效。常见错误是把权重放在结构体第二个字段却没自定义排序规则,结果算法从最大边开始选,根本得不到最小生成树。

推荐统一用 std::vector<:tuple int>> 存边:权重在前,sort 自然按权重升序;或者用结构体 + lambda:

std::vector edges;
std::sort(edges.begin(), edges.end(), [](const Edge& a, const Edge& b) {
    return a.weight < b.weight;
});

并查集(Union-Find)不路径压缩会超时吗?

会。尤其点数 > 1e4 时,朴素 find 可能退化成 O(n) 单次查询,整体复杂度接近 O(mn),Kruskal 就失去意义。必须做路径压缩 + 按秩合并(或只做路径压缩也够大多数 OJ 场景)。

关键实现要点:

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

  • parent[i] = i 初始化不能漏
  • find 中递归调用后要赋值更新:return parent[x] = find(parent[x])
  • unionSet 里判断的是 rootA != rootB,不是 a != b
struct UnionFind {
    std::vector parent, rank;
    UnionFind(int n) : parent(n), rank(n, 0) {
        for (int i = 0; i < n; ++i) parent[i] = i;
    }
    int find(int x) {
        return parent[x] == x ? x : parent[x] = find(parent[x]);
    }
    bool unionSet(int a, int b) {
        int ra = find(a), rb = find(b);
        if (ra == rb) return false;
        if (rank[ra] < rank[rb]) std::swap(ra, rb);
        parent[rb] = ra;
        if (rank[ra] == rank[rb]) rank[ra]++;
        return true;
    }
};

输入边时顶点编号从 0 还是 1 开始?

取决于题干。国内 OJ(如洛谷、PTA)多数用 1-indexed 顶点,但 UnionFind 类内部下标是 0-based,所以读入后立刻减 1,别等到排序或合并时再处理——容易忘、易出界、调试困难。

例如输入 u v w,统一转成:

Elai.io
Elai.io

AI视频生成工具,可以使用文字、URL、PDF等生成视频

下载
int u, v, w;
std::cin >> u >> v >> w;
edges.emplace_back(w, u - 1, v - 1); // 权重前置,顶点归零

后续所有 unionSet(u-1, v-1)find(u-1) 都基于 0-indexed,逻辑干净。

如何验证 Kruskal 输出的边确实构成 MST?

最简验证方式:检查是否恰好选了 n - 1 条边,且总权重与标准答案一致。更稳妥的做法是,在每次 unionSet 成功后记录该边,并确保最终 result.size() == n - 1;若提前结束(边用完但没凑够),说明图不连通——这时应返回错误或输出 -1,而不是强行输出部分结果。

典型收尾逻辑:

int cnt = 0, totalWeight = 0;
for (auto [w, u, v] : edges) {
    if (uf.unionSet(u, v)) {
        totalWeight += w;
        cnt++;
        if (cnt == n - 1) break;
    }
}
if (cnt != n - 1) {
    std::cout << "-1\n"; // 不连通
} else {
    std::cout << totalWeight << "\n";
}

注意:不要在循环外再查 uf.find(0) == uf.find(i) 判断连通性——Kruskal 过程中已隐含该信息,重复检查无必要且易错。

真正容易被忽略的是:当存在多条相同权重的边时,sort 的稳定性不影响正确性,但不同实现可能选出不同的 MST(只要总权相等就合法)。如果题目要求输出字典序最小的边集,就得在权重相同时按端点二次排序——这点几乎没人提,但现场笔试真会卡。

相关文章

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
sort排序函数用法
sort排序函数用法

sort排序函数的用法:1、对列表进行排序,默认情况下,sort函数按升序排序,因此最终输出的结果是按从小到大的顺序排列的;2、对元组进行排序,默认情况下,sort函数按元素的大小进行排序,因此最终输出的结果是按从小到大的顺序排列的;3、对字典进行排序,由于字典是无序的,因此排序后的结果仍然是原来的字典,使用一个lambda表达式作为key参数的值,用于指定排序的依据。

381

2023.09.04

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

194

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

186

2025.07.04

c语言union的用法
c语言union的用法

c语言union的用法是一种特殊的数据类型,它允许在相同的内存位置存储不同的数据类型,union的使用可以帮助我们节省内存空间,并且可以方便地在不同的数据类型之间进行转换。使用union时需要注意对应的成员是有效的,并且只能同时访问一个成员。本专题为大家提供union相关的文章、下载、课程内容,供大家免费下载体验。

122

2023.09.27

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

202

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

189

2025.11.08

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

391

2023.08.14

php代码编辑器入口汇总
php代码编辑器入口汇总

本文整理了主流PHP代码编辑器的官网入口及在线使用链接,阅读专题下面的文章了解更多详细内容。

26

2026.01.04

php代码编辑器地址汇总
php代码编辑器地址汇总

本文整理了主流PHP代码编辑器的官网入口及在线使用链接,阅读专题下面的文章了解更多详细内容。

0

2026.01.04

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 6万人学习

C 教程
C 教程

共75课时 | 3.9万人学习

C++教程
C++教程

共115课时 | 11.1万人学习

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

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