2021-08-10校网比赛B题

本文介绍了一种解决最小生成树问题的方法——Kruskal算法,通过建立超级源点并进行边的排序,逐步连接节点,直至形成一棵最小生成树。在实现过程中需要注意避免超时和正确存储边。代码示例展示了如何应用Kruskal算法来找到最小的雇佣和交流费用总和。
摘要由CSDN通过智能技术生成

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

思路:
建立一个超级源点S,
S向所有人连边,边权为雇佣费用
人与人之间存在连边,即交流费用
求最小生成树即可
注意几个细节:

  1. 对于最小生成树的边存储只能存一半,否则会超时。
  2. 千万不要打快读,我也不知道为什么比赛几个打快读的都炸掉了
#include <cstdio>
#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;

//typedef long long int;

const int N = 1e6 + 10;
const int M = 1e3 + 10;
struct node {int u, v, w;} e[N];
int n, cnt, ans, t, f[N], a, tot;

bool cmp(node a, node b) {return a.w < b.w;}

int find(int x) {return (x == f[x]) ? x : (f[x] = find(f[x]));}

void unionn(int x, int y, int w) {f[x] = y; ans += w; tot++;}

int main()
{
	scanf("%d", &n);
	for(int i = 1; i <= n; i++)
	 for(int j = 1; j <= n; j++)
	 {
	 	scanf("%d", &t);
		if(i < j) e[++cnt] = (node){i, j, t};	
	 }
	for(int i = 1; i <= n; i++) 
		scanf("%d", &a), f[i] = i, e[++cnt] = (node){0, i, a};
	sort(e + 1, e + 1 + cnt, cmp);
	for(int i = 1; i <= cnt; i++)
	{
		int u = e[i].u, v = e[i].v;
		int fx = find(u), fy = find(v);
		if(fx != fy) unionn(fx, fy, e[i].w);
		if(tot == n) break;
	}
	printf("%d\n", ans);
	return 0;
} 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值