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

P5633 最小度限制生成树

题目

P5633 最小度限制生成树

算法标签: 最小生成树, k r u s k a l kruskal kruskal重构树, 贪心

思路

题目大意就是构建最小生成树, 但是要求点 s s s d e g deg deg必须是 k k k

  1. 首先先将与 s s s连接的边先删除, 将与 s s s连接的边标记为特殊边
  2. 然后对剩余的部分进行 k r u s k a l kruskal kruskal算法, 这样就会产生很多连通块, 在合并两个连通块的时候将特殊边权更大的集合合并到特殊边更小的集合, 这是个贪心的策略, 保证最后连接 s s s边权尽可能小
  3. 然后在合并两个集合的过程中, 如果发现存在特殊边, 那么计算如果使用普通边, 会节约多少的值
  4. 然后枚举所有的特殊边, 因为要求 d e g = k deg = k deg=k, 计算在合并连通块过程中使用了多少条特殊边
  5. 然后对 k − d e g k- deg kdeg条特殊边从小到大排序, 选出最优的 k − d e g k - deg kdeg条边

算法时间复杂度 O ( m log ⁡ m ) O(m \log m) O(mlogm)

代码

#include <iostream>
#include <algorithm>
#include <cstring>
#include <vector>using namespace std;const int N = 50010, M = 500010, INF = 0x3f3f3f3f;int n, m, s, k, idx;struct Edge {int u, v, w;bool operator<(const Edge &e) const {return w < e.w;}
} edges[M << 1];
int p[N], val[N];void add(int u, int v, int w) {edges[idx++] = {u, v, w};
}int find(int x) {if (p[x] != x) p[x] = find(p[x]);return p[x];
}int kruskal() {for (int i = 1; i <= n; ++i) p[i] = i;sort(edges, edges + idx);int ans = 0;vector<int> vec;// 构建最小生成树for (int i = 0; i < idx; ++i) {auto [u, v, w] = edges[i];int fu = find(u), fv = find(v);if (fu == fv) continue;// 总是将较大val的集合合并到较小val的集合if (val[fu] > val[fv]) swap(fu, fv);p[fv] = fu;ans += w;if (val[fv] < INF) vec.push_back(val[fv] - w);}// 计算必须选择的特殊边int deg = 0;for (int i = 1; i <= n; ++i) {if (p[i] != i || i == s) continue;if (val[i] >= INF) {cout << "Impossible" << "\n";exit(0);}deg++;ans += val[i];}// 检查可行性if (deg > k || deg + vec.size() < k) {cout << "Impossible" << "\n";exit(0);}sort(vec.begin(), vec.end());for (int i = 0; i < k - deg; ++i) ans += vec[i];return ans;
}int main() {ios::sync_with_stdio(false);cin.tie(0), cout.tie(0);memset(val, 0x3f, sizeof val);cin >> n >> m >> s >> k;// 处理输入for (int i = 0; i < m; ++i) {int u, v, w;cin >> u >> v >> w;if (u == s) val[v] = min(val[v], w);else if (v == s) val[u] = min(val[u], w);else add(u, v, w);}// 特殊情况处理:如果k=0但s有边if (k == 0) {for (int i = 1; i <= n; ++i) {if (val[i] < INF) {cout << "Impossible" << "\n";return 0;}}}int ans = kruskal();cout << ans << "\n";return 0;
}
http://www.xdnf.cn/news/220519.html

相关文章:

  • Linux环境变量以及进程虚拟地址原理
  • DVWA靶场保姆级通关教程---02命令注入
  • 5.4.2 MVVM例2-用户控件的使用(水在水管中流动的实例)
  • 路径规划算法总结:从 Dijkstra 到 A* 与 Hybrid A
  • GUI_DrawPixel 函数详解
  • BalenaEtcher 2.1镜像烧录工具软件下载及安装教程
  • Vite性能优化指南 ✅
  • 强化学习(二)马尔科夫决策过程(MDP)
  • java AsyncTool
  • ACTF2025 - WEB Excellent-Site
  • 第十章:CrewAI - 面向流程的多 Agent 结构化协作
  • Andorid车机UI适配,AndroidUI图px的单位,如何适配1920x720,PPI100的屏幕设备
  • 【GESP】C++三级练习 luogu-B2117 整理药名
  • Rockchip Android平台打开GKI无法开机问题
  • 应用服务器-IIS
  • 推荐系统中 Label 回收机制之【时间窗口设计】
  • 基于Lucene的多场景检索系统开发指南
  • [按键安卓ios脚本辅助插件开发]数组排序函数例子
  • 明远智睿SSD2351开发板:开启嵌入式开发新篇程
  • C#实现对达索(Dassault)SolidWorks中3D图纸转化为手机可直接查看预览图纸格式
  • 高级项目管理
  • 巧记英语四级单词 Unit6-下【晓艳老师版】
  • C++程序退出时的对象析构陷阱:深度解析与避坑指南
  • mysql 事务中如果有sql语句出错,会导致自动回滚吗?
  • 力扣刷题总表
  • 【Vue】 实现TodoList案例(待办事项)
  • Java高频面试之并发编程-10
  • C++之string
  • 如何在本地部署小智服务器:从源码到全模块运行的详细步骤
  • CA校验主辅小区配置及UE能力