【N - 畅通工程再续】

80 篇文章 0 订阅
80 篇文章 0 订阅

思路:

  • 最小生成树Prim。要注意即使是 double,闭区间也是必须写成闭区间的 ( >= )

代码:

  • 46ms 1532kB
//46ms		1532kB


#include <iostream>
#include <queue>
#include <cstring>
#include <cmath>

using namespace std;

const int maxn = 105;

int N;
bool ok;
double ans;
double dis[maxn];
bool vis[maxn];
double mp[maxn][maxn];
struct ORD{
	int x,y;
}ord[maxn];
struct NODE{
	int id;
	double dis;
	friend bool operator > (NODE a, NODE b){
		return a.dis > b.dis;
	}
	NODE(int id,double dis) : id(id) , dis(dis) {} ;
};
priority_queue<NODE , vector<NODE> , greater<NODE> >Q;

void INIT(){
	ans = 0;
	ok = true;
	for(int i=0;i<maxn;i++){
		dis[i] = 1500;
	}
	memset(vis,0,sizeof(vis));
	return ;
}

void PRIM(){
	Q.push(NODE(1,0)) ; dis[1]=0 ;
	while(Q.size()){
		NODE cur = Q.top() ; Q.pop() ;
		int id = cur.id ;
		if(vis[id])
			continue;
		vis[id] = true;
		ans += dis[id];
		for(int i=1;i<=N;i++)
			if(!vis[i] && dis[i] > mp[id][i]){
				dis[i] = mp[id][i];
				Q.push(NODE(i , dis[i]));
			}
	}
	for(int i=1;i<=N;i++)
		if(!vis[i]){
			ok = false;
			break;
		}
	return ;
}

int main(){
	int T;cin>>T;
	while(T--){
		INIT();
		cin>>N;
		for(int i=1;i<=N;i++)
			cin>>ord[i].x>>ord[i].y;
		for(int i=1;i<=N;i++){
			for(int j=i+1;j<=N;j++){
				double w = sqrt((ord[i].x - ord[j].x)*(ord[i].x - ord[j].x) + (ord[i].y - ord[j].y)*(ord[i].y - ord[j].y)) ;
				if(w >= 10 && w <= 1000)
					//double也要写 ‘=’
					mp[i][j] = mp[j][i] = w;
				else
					mp[i][j] = mp[j][i] = 2000;//绝不允许建路,故应 > 1500 
			}
			mp[i][i] = 0;
		}
		PRIM();
		if(ok)
			printf("%.1f\n" , 100*ans);
		else
			cout<<"oh!"<<endl;
	}
	return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值