题目大意:
国防部想在北部的前哨之间建立一个无线网络连接这些前哨。在建立网络时使用了两种不同的通信技术:每个前哨有一个无线电收发器,有一些前哨还有一个卫星频道。
任何两个拥有卫星频道的前哨之间可以直接通过卫星进行通信,而且卫星通信跟距离和位置无关。否则,两个前哨之间通过无线电收发器进行通信,并且这两个前哨之间的距离不能超过D,这个D值取决于无线电收发器的功率。功率越大,D值也就越大,但是价格也越高。出于对购买费用和维护费用的考虑,所有使用无线电接收器进行通信的前哨的收发器必须相同,即D值相同。
你的任务是计算无线电收发器D值的最小值,且保证每两个前哨之间至少有一条通信线路(直接或间接地连接这两个前哨)。
本题就是,求出最小生成树的时候,顺便记录生成树的权值,然后从小到大排序,最后sum-m既是最终结果。
#include <iostream>
#include <cstring>
#include <iomanip>
#include <algorithm>
#include <cmath>
#include <cstdio>
using namespace std;
const int MAXN = 510;
const int INF = 1000010;
int n, m;
int Edge[MAXN][MAXN];
void Inite()
{
int i, j;
int X[MAXN], Y[MAXN];
memset(Edge, 0, sizeof(Edge));
cin >> m >> n;
for(i = 0; i < n; ++i)
cin >> X[i] >> Y[i];
for (i = 0; i < n; ++i)
{
for(j = i+1; j < n; ++j)
{
Edge[i][j] = (X[i] - X[j]) * (X[i] - X[j]) + (Y[i] - Y[j]) * (Y[i] - Y[j]);
Edge[j][i] = Edge[i][j];
}
}
for (i = 0; i < n; ++i)
{
for (j = 0; j < n; ++j)
{
if(Edge[i][j] == 0)
Edge[i][j] = INF;
}
}
}
void Prim()
{
int lowcost[MAXN];
int dist[MAXN];
int sum = 0;
int i, j;
for (i = 0; i < n; ++i)
{
lowcost[i] = Edge[0][i];
}
lowcost[0] = -1;
memset(dist, 0, sizeof(dist));
for(i = 1; i < n; ++i)
{
int MIN = INF;
int v;
for(j = 0; j < n; ++j)
{
if (lowcost[j] != -1 && lowcost[j] < MIN)
{
v = j;
MIN = lowcost[j];
}
}
dist[sum++] = MIN;
lowcost[v] = -1;
for (j = 0; j < n; ++j)
{
if (lowcost[j] != -1 && lowcost[j] > Edge[v][j])
{
lowcost[j] = Edge[v][j];
}
}
}
sort(dist, dist+sum);
cout << setiosflags(ios::fixed) << setprecision(2) <<sqrt((double)dist[ sum-m ])<<endl;
}
int main()
{
int T;
cin>>T;
while (T--)
{
Inite();
Prim();
}
return 0;
}