[特殊字符] 蓝桥杯 Java B 组 之最小生成树(Prim、Kruskal) 并查集应用

ops/2025/2/28 13:35:44/

Day 3:最小生成树(Prim、Kruskal) & 并查集应用


📖 一、最小生成树(MST)简介

最小生成树(Minimum Spanning Tree, MST) 是一个 无向连通图最小代价子图,它包含 所有节点,且边的权重总和最小。

📌 最小生成树的性质

  1. 连通性:必须包含所有节点,且保证是连通的。
  2. 边数 = 节点数 - 1:MST 的边数始终是 V - 1V 是顶点数)。
  3. 权值最小:MST 的边权和最小。

📌 最小生成树的常见算法

算法核心思想时间复杂度
Kruskal贪心 + 并查集,按 边权 递增排序,逐步合并连通分量O(E log E)(边排序)
Prim贪心 + 最小堆,按 最小权重 逐步扩展 MSTO(E log V)(优先队列优化)

Kruskal 适用于稀疏图,Prim 适用于稠密图


📖 二、Kruskal 算法(贪心 + 并查集)

适用范围

  • 边稀疏的图(E 边较少)。
  • 适用于 离散集问题(如岛屿连通、网络电缆)。

🔹 1. 题目:连接所有城市的最低成本(Leetcode 1135)

题目描述: 给定 n 个城市,connections[i] = (u, v, cost) 表示 u-v 之间有条代价 cost 的边。找到最小代价的连接方案,使得所有城市连通,否则返回 -1

示例

输入: n = 3, connections = [[1,2,5],[1,3,6],[2,3,2]]
输出: 7 (选 [1,2] 和 [2,3])

🔹 2. 思路

  1. 贪心 + Kruskal 算法
    • 按边权排序(从小到大)。
    • 使用并查集(Union-Find) 逐步合并两个城市(避免形成环)。
    • n-1 条边后停止,若无法连通所有城市,则返回 -1

🔹 3. 代码实现(Kruskal)

import java.util.*;public class KruskalMST {static class UnionFind {int[] parent, rank;public UnionFind(int n) {parent = new int[n + 1];rank = new int[n + 1];for (int i = 0; i <= n; i++) parent[i] = i;}public int find(int x) {if (x != parent[x]) parent[x] = find(parent[x]); // 路径压缩return parent[x];}public boolean union(int x, int y) {int rootX = find(x), rootY = find(y);if (rootX == rootY) return false; // 已经连通if (rank[rootX] > rank[rootY]) parent[rootY] = rootX;else if (rank[rootX] < rank[rootY]) parent[rootX] = rootY;else {parent[rootY] = rootX;rank[rootX]++;}return true;}}public int minCostToConnectCities(int n, int[][] connections) {Arrays.sort(connections, Comparator.comparingInt(a -> a[2])); // 按边权排序UnionFind uf = new UnionFind(n);int totalCost = 0, edgesUsed = 0;for (int[] conn : connections) {if (uf.union(conn[0], conn[1])) { // 连接成功totalCost += conn[2];edgesUsed++;if (edgesUsed == n - 1) break; // 最小生成树完成}}return edgesUsed == n - 1 ? totalCost : -1; // 判断是否连通}public static void main(String[] args) {KruskalMST solution = new KruskalMST();int[][] connections = {{1,2,5}, {1,3,6}, {2,3,2}};int n = 3;System.out.println(solution.minCostToConnectCities(n, connections)); // 输出 7}
}

🔹 4. 代码讲解

  1. 并查集(Union-Find)
    • find(x): 查找根节点(带路径压缩)。
    • union(x, y): 合并两个集合(按秩合并)。
  2. Kruskal 算法
    • 排序:按边权递增排序。
    • 遍历边:检查 u-v 是否已经连通,若未连通,则加入 MST。
    • 判断最终连通性:若选出的边数为 n-1,返回总代价,否则返回 -1

✅ 时间复杂度O(E log E)(排序 + 并查集)


📖 三、Prim 算法(贪心 + 最小堆)

适用范围

  • 边稠密的图(E 边较多)。
  • 适用于 邻接矩阵/邻接表存储

🔹 1. 代码实现(Prim)

import java.util.*;public class PrimMST {public int minCostToConnectCities(int n, int[][] connections) {Map<Integer, List<int[]>> graph = new HashMap<>();for (int[] edge : connections) {graph.putIfAbsent(edge[0], new ArrayList<>());graph.putIfAbsent(edge[1], new ArrayList<>());graph.get(edge[0]).add(new int[]{edge[1], edge[2]});graph.get(edge[1]).add(new int[]{edge[0], edge[2]});}PriorityQueue<int[]> pq = new PriorityQueue<>(Comparator.comparingInt(a -> a[1]));pq.offer(new int[]{1, 0}); // 从城市 1 开始Set<Integer> visited = new HashSet<>();int totalCost = 0;while (!pq.isEmpty() && visited.size() < n) {int[] cur = pq.poll();int city = cur[0], cost = cur[1];if (visited.contains(city)) continue; // 已访问visited.add(city);totalCost += cost;for (int[] neighbor : graph.getOrDefault(city, new ArrayList<>())) {if (!visited.contains(neighbor[0])) {pq.offer(neighbor);}}}return visited.size() == n ? totalCost : -1;}public static void main(String[] args) {PrimMST solution = new PrimMST();int[][] connections = {{1,2,5}, {1,3,6}, {2,3,2}};int n = 3;System.out.println(solution.minCostToConnectCities(n, connections)); // 输出 7}
}

📖 四、Kruskal vs Prim

算法核心数据结构适用场景时间复杂度
Kruskal并查集稀疏图(边少)O(E log E)
Prim最小堆(优先队列)稠密图(边多)O(E log V)

📖 五、总结

🎯 掌握 Kruskal(贪心 + 并查集),适用于离散集合最小代价连接问题
🎯 掌握 Prim(贪心 + 最小堆),适用于稠密图的最小生成树问题
🎯 最小生成树的应用

  • 网络连接(最小代价连通所有城市)
  • 电网铺设(最少电缆铺设)
  • 地图导航(最短成本的道路规划)


http://www.ppmy.cn/ops/161972.html

相关文章

倚光科技:助力玻璃非球面的打样与小批量生产

在现代光学和精密制造领域&#xff0c;非球面光学元件凭借其卓越的光学性能&#xff0c;已成为推动高端科技发展的核心组件。相比于传统的球面透镜&#xff0c;非球面透镜能够显著减少光学系统中的像差和畸变&#xff0c;大幅提升成像质量、系统紧凑性和能量利用率。因此&#…

【JavaScript】《JavaScript高级程序设计 (第4版) 》笔记-Chapter25-客户端存储

二十五、客户端存储 客户端存储 随着 Web 应用程序的出现&#xff0c;直接在客户端存储用户信息的需求也随之出现。这背后的想法是合理的&#xff1a;与特定用户相关的信息应该保存在用户的机器上。无论是登录信息、个人偏好&#xff0c;还是其他数据&#xff0c;Web 应用程序提…

【读书笔记·VLSI电路设计方法解密】问题57:逻辑合成过程中插入测试的目的是什么

如第3章第20题所述&#xff0c;可测试性设计&#xff08;Design for Testability, DFT&#xff09;是创建具有商业价值的产品时需要考虑的一个非常重要的问题。为了实现DFT功能&#xff0c;使设计能够检测制造缺陷&#xff0c;需要在设计中添加额外的测试电路&#xff0c;而这些…

想学python进来看看把

目录 什么是python 我将列举python与其他几种编程语言的对比 Python vs Java Python vs JavaScript Python vs C​编辑 我将列举代码示例帮大家来理解 python c/c java 写一个python程序 你一定要知道什么是BUG呦 遇到bug怎么办 1. 保持冷静 2. 重现 Bug 3. 阅…

Spring Boot 3 集成 RabbitMQ 实践指南

Spring Boot 3 集成 RabbitMQ 实践指南 1. RabbitMQ 核心原理 1.1 什么是RabbitMQ RabbitMQ是一个开源的消息代理和队列服务器&#xff0c;使用Erlang语言开发&#xff0c;基于AMQP&#xff08;Advanced Message Queuing Protocol&#xff09;协议实现。它支持多种消息传递模…

【FL0086】基于SSM和微信小程序的垃圾分类小程序

&#x1f9d1;‍&#x1f4bb;博主介绍&#x1f9d1;‍&#x1f4bb; 全网粉丝10W,CSDN全栈领域优质创作者&#xff0c;博客之星、掘金/知乎/b站/华为云/阿里云等平台优质作者、专注于Java、小程序/APP、python、大数据等技术领域和毕业项目实战&#xff0c;以及程序定制化开发…

etcd 3.15 三节点集群管理指南

本文档旨在提供 etcd 3.15 版本的三节点集群管理指南&#xff0c;涵盖节点的新增、删除、状态检查、数据库备份和恢复等操作。 1. 环境准备 1.1 系统要求 操作系统&#xff1a;Linux&#xff08;推荐 Ubuntu 18.04 或 CentOS 7&#xff09; 内存&#xff1a;至少 2GB 磁盘&a…

编写第一个 C++ 程序 – Hello World 示例

“Hello World”程序是学习任何编程语言的第一步&#xff0c;也是您将学习的最直接的程序之一。它是用于演示编码过程如何工作的基本程序。您所要做的就是在输出屏幕上显示 “Hello World”。 C Hello World 程序 下面是在控制台屏幕上打印 “Hello World” 的 C 程序。 // …