本文已参与「新人创作礼」活动,一起开启掘金创作之路。
异想之旅:本人原创博客完全手敲,绝对非搬运,全网不可能有重复;本人无团队,仅为技术爱好者进行分享,所有内容不牵扯广告。本人所有文章仅在CSDN、掘金和个人博客(一定是异想之旅域名)发布,除此之外全部是盗文!
本题同时存在于,除输入数据的输入顺序外无任何区别,此处使用题目描述更详细的一本通的原题,洛谷题目思路完全相同,代码唯一区别会在注释中标明。
1487:【例 2】北极通讯网络
题目描述
原题来自:Waterloo University 2002
北极的某区域共有 座村庄,每座村庄的坐标用一对整数 () 表示。为了加强联系,决定在村庄之间建立通讯网络。通讯工具可以是无线电收发机,也可以是卫星设备。所有的村庄都可以拥有一部无线电收发机, 且所有的无线电收发机型号相同。但卫星设备数量有限,只能给一部分村庄配备卫星设备。
不同型号的无线电收发机有一个不同的参数 ,两座村庄之间的距离如果不超过 就可以用该型号的无线电收发机直接通讯, 值越大的型号价格越贵。拥有卫星设备的两座村庄无论相距多远都可以直接通讯。\n现在有 台卫星设备,请你编一个程序,计算出应该如何分配这 台卫星设备,才能使所拥有的无线电收发机的 值最小,并保证每两座村庄之间都可以直接或间接地通讯。
例如,对于下面三座村庄:
其中
如果没有任何卫星设备或只有 台卫星设备 ( 或 ),则满足条件的最小的 ,因为 和 , 和 可以用无线电直接通讯;而 和 可以用 中转实现间接通讯 (即消息从 传到 ,再从 传到 );
如果有 台卫星设备 (),则可以把这两台设备分别分配给 和 ,这样最小的 可取 ,因为 和 之间可以用无线电直接通讯; 和 之间可以用卫星直接通讯; 和 可以用 中转实现间接通讯。
如果有 台卫星设备,则 两两之间都可以直接用卫星通讯,最小的 可取 。
输入格式
第一行为由空格隔开的两个整数 ;\n第 行,每行两个整数,第 行的 表示第 座村庄的坐标 ()。
输出格式
一个实数,表示最小的 值,结果保留 位小数。
输入输出样例
In 1:
3 2
10 10
10 0
30 0
Out 1:
10.00
数据范围
对于全部数据,。
题解
水题,不知道为什么花了我这么久时间
我一定没有摸鱼
显然,当卫星设备数为 或 时,本题退化为最小生成树求最长边,这个大家应该都可以理解的对吧。
那么,如果这时候有 套卫星设备会怎样?这是比排队接水还简单的贪心了吧,就是把最小生成树中的最长边去掉,这个边连接的两个村庄之间用卫星设备,答案也就变成了最小生成树中第二长的边。
可以总结规律了。 台卫星设备可以连接 个最小生成树为一个整体,那么也就是说整个最小生成树可以少 条边。少哪 条边呢?自然是最长的 条边。这样子一来,答案就是最小生成树中第 条边的长度,其中 就是最小生成树中边的个数。
由于题目数据实在是太水了, 才有 ,完全没必要考虑堆优化。直接用裸模板就行。
#include <bits/stdc++.h>
using namespace std;
int n, k, m[501][2];
struct Edge {
double f, t, w;
} edge[250000], edges[501]; // edge是所有的边,edges记录所有加入了最小生成树的边
bool cmp(Edge a, Edge b) { return a.w < b.w; }
class bcj { // 用类实现并查集可以省去全局变量冲突的烦恼,看起来更清爽
public:
int f[250000];
void init(int n) {
for (int i = 1; i <= n; i++) {
f[i] = i;
}
}
int find(int x) {
while (f[x] != x) {
f[x] = f[f[x]];
x = f[x];
}
return x;
}
void merge(int x, int y) {
int fx = find(x), fy = find(y);
if (fx != fy) {
f[fx] = fy;
}
}
};
int main() {
cin >> n >> k;
// cin >> k >> n; // 提交洛谷用这一行
for (int i = 1; i <= n; i++) {
cin >> m[i][0] >> m[i][1];
}
int cnt = 0;
for (int i = 1; i <= n; i++) {
for (int j = i + 1; j <= n; j++) {
edge[++cnt].f = i;
edge[cnt].t = j;
edge[cnt].w =
sqrt(pow(m[i][0] - m[j][0], 2) + pow(m[i][1] - m[j][1], 2));
}
}
sort(edge + 1, edge + cnt + 1, cmp);
bcj b;
b.init(n);
int cntt = 0;
for (int i = 1; i <= cnt && cntt < n; i++) {
if (b.find(edge[i].f) != b.find(edge[i].t)) {
b.merge(edge[i].f, edge[i].t);
edges[cntt++] = edge[i];
}
}
if (k <= 1) {
printf("%.2f\n", edges[cntt - 1].w);
} else {
printf("%.2f\n", edges[cntt - k].w);
}
return 0;
}