CCF CSP 201812-4数据中心【二分答案】【最小生成树】

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

问题分析

抛开特别长的题干,我们提取关键信息,发现:是一个无向图,边权,求最优树结构。

最小化传输时间Tmax
Tmax=max{Th} Th=max{th,i} Tmax=max{thi} 1≤h ≤dep,1 ≤i ≤size(h)
Tmax=max{T1,T2,T3}
=max{t1,1,t1,2,t1,3,t2,1,t2,2,t2,3, ,t3,1}
=max{树上边权}
问题转化为给定一个无向图,求解一棵生成树,使最大边权最小。
二分解法
 “求解一颗生成树,使得最大边权最小!”  最大值最小,很容易想到二分
 二分答案 ans,问题转化为原图中不大于 ans 的边能否使原图联通
 二分答案复杂度O(logK),判断连通性复杂度O(N+M)
 总复杂度O((N+M)logK)  100分 
最小瓶颈生成树
 利用性质:最小生成树一定是瓶颈生成树
 求 MST 输出最大的边
我是利用的最小生成树,参考代码如下:

#include<stdio.h>
#include<algorithm>
using namespace std;
struct edge{
	int e1,e2,w;
	bool operator<(const edge&x){return w<x.w;}
}a[91000];
int fa[310];
int find(int x){return fa[x]==x?x:fa[x]=find(fa[x]);}
int Kruskal(int n){
	int E=0,i=0,ans=0;
	for(int i=0;i<=n;i++)fa[i]=i;
	while(E<n){
		int u = find(a[i].e1), v = find(a[i].e2);
		if (u != v){
			fa[u] = v;
			E++;
			ans+=a[i].w;
		}
		i++;
	}
	return ans;	
}
int main(){
	int n,e,x;
	scanf("%d",&n);
	for(int i=0;i<n;i++){
		scanf("%d",&x);
		a[i].e1=i+1;
		a[i].e2=0;
		a[i].w=x;
	}
	e=n;
	for(int i=1;i<=n;i++)
		for(int j=1;j<=n;j++){
			scanf("%d",&x);
			if(i<j){
				a[e].e1=i;
				a[e].e2=j;
				a[e].w=x;
				e++;
			}
		}
	sort(a,a+e);
	printf("%d",Kruskal(n));
	return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值