博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
紫书 习题11-11 UVa 1644 (并查集)
阅读量:6470 次
发布时间:2019-06-23

本文共 1077 字,大约阅读时间需要 3 分钟。

这道题感觉思路非常巧妙, 我是看了别人的博客才想明白的。

这里用到了并查集, 以根节点为中心城市, 然后把边从大到小排序, 每次的当前的边即为容量,

因为是目前的最小值, 然后去算总的容量, 每次选容量大的点作为新的根节点, 也就是新的

中心城市。细节见代码。

#include
#include
#define REP(i, a, b) for(int i = (a); i < (b); i++)using namespace std;typedef long long ll;const int MAXN = 212345;struct node{ ll u, v, w; bool operator < (const node& rhs) const { return w > rhs.w; }}Edge[MAXN];ll sum[MAXN], cnt[MAXN];int f[MAXN], n;int find(int x){ if(f[x] != x) f[x] = find(f[x]); return f[x];}int main(){ while(~scanf("%d", &n)) { REP(i, 1, n + 1) sum[i] = 0, cnt[i] = 1, f[i] = i; REP(i, 0, n - 1) scanf("%lld%lld%lld", &Edge[i].u, &Edge[i].v, &Edge[i].w); sort(Edge, Edge + n - 1); ll ans = 0; REP(i, 0, n - 1) { int u = find(Edge[i].u), v = find(Edge[i].v), w = Edge[i].w; ll sumu = sum[u] + cnt[v] * w, sumv = sum[v] + cnt[u] * w; if(sumu < sumv) f[u] = v, ans = sumv, cnt[v] += cnt[u], sum[v] = sumv; else f[v] = u, ans = sumu, cnt[u] += cnt[v], sum[u] = sumu; } printf("%lld\n", ans); } return 0;}

转载于:https://www.cnblogs.com/sugewud/p/9819533.html

你可能感兴趣的文章
spring boot configuration annotation processor not found in classpath问题解决
查看>>
由中序遍历和后序遍历求前序遍历
查看>>
我学习参考的网址
查看>>
[Processing]点到线段的最小距离
查看>>
GitHub使用教程、注册与安装
查看>>
<<The C Programming Language>>讀書筆記
查看>>
git代码冲突
查看>>
解析查询 queryString 请求参数的函数
查看>>
学生选课系统数据存文件
查看>>
git bash 风格调整
查看>>
linux操作系统加固软件,系统安全:教你Linux操作系统的安全加固
查看>>
linux中yum源安装dhcp,24.Linux系统下动态网络源部署方法(dhcpd)
查看>>
C#技术------垃圾回收机制(GC)
查看>>
漫谈并发编程(三):共享受限资源
查看>>
【转】github如何删除一个仓库
查看>>
HDOJ-1010 Tempter of the Bone
查看>>
JavaNIO基础02-缓存区基础
查看>>
日本开设无人机专业,打造无人机“人才市场”
查看>>
190行代码实现mvvm模式
查看>>
兼容几乎所有浏览器的透明背景效果
查看>>