【NOIP2015模拟10.22】最小代价题解

该博客介绍了一道关于无向带权图的问题,目标是在保持白点到最近黑点最短距离不变的前提下,寻找最小代价的连接方案。通过新建源点连接所有黑点,跑最短路算法,再利用贪心策略选择每条白点到黑点的最短边,以求得最小代价。
摘要由CSDN通过智能技术生成

Description

给出一幅由n个点m条边构成的无向带权图。
其中有些点是黑点,其他点是白点。
现在每个白点都要与他距离最近的黑点通过最短路连接(如果有很多个黑点,可以选取其中任意一个),我们想要使得花费的代价最小。请问这个最小代价是多少?
注意:最后选出的边保证每个白点到离它最近的黑点的距离仍然等于原图中的最短距离。

Input

第一行两个整数n,m;
第二行n 个整数,0表示白点,1 表示黑点;
接下来m 行,每行三个整数x,y,z,表示一条连接x和y 点,权值为z 的边。

Output

如果无解,输出impossible;
否则,输出最小代价。

Sample Input

5 7
0 1 0 1 0
1 2 11
1 3 1
1 5 17
2 3 1
3 5 18
4 5 3
2 4 5

Sample Output

5
【样例解释】
选 2、4、6三条边

Data constraint

对30%的输入数据: 1≤n≤10, 1≤m≤20;
对100%的输入数据:1≤n≤100000,1≤m≤200000,1≤z≤1000000000

Solution

首先,我们新建一个源点,连接所有黑点,边权为0,然后跑一遍最短路,即可求出每一个白点到黑点的最短距离。
然后,利用贪心的思想。我们扫一遍每一个白点,在扫一遍每一条直接连向它的边。对于一条边,如果是从当前的白点走过去,也就是dis[i]==dis[j]+len那么我们就要找到一条最短的这样的边,并加入答案,其它长的边全部删掉。如果是从对面走到当前这一个白点,则不做任何操作。

Code

#include<cstdio>
#include<cstring>
#include<queue>
#include<iostream>
#define ll long long
using namespace std;
const int N=100001;
int n,m,cnt=0,sum=0,sum1=0;
bool a[N],vis[N];
int head[N],b[N],white[N];
ll dis[N],ans=0;
struct node {
   
	int to,next,len;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值