zoj 1914 poj 2349 Arctic Network

题目大意:

    国防部想在北部的前哨之间建立一个无线网络连接这些前哨。在建立网络时使用了两种不同的通信技术:每个前哨有一个无线电收发器,有一些前哨还有一个卫星频道。
    任何两个拥有卫星频道的前哨之间可以直接通过卫星进行通信,而且卫星通信跟距离和位置无关。否则,两个前哨之间通过无线电收发器进行通信,并且这两个前哨之间的距离不能超过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;
}


你可能感兴趣的:(MST)