2439 四叶草魔杖
问题描述
魔杖护法Freda融合了四件武器,于是魔杖顶端缓缓地生出了一棵四叶草,四片叶子幻发着淡淡的七色光。圣剑护法rainbow取出了一个圆盘,圆盘上镶嵌着N颗宝石,编号为0~N-1。第i颗宝石的能量是Ai。如果Ai>0,表示这颗宝石能量过高,需要把Ai的能量传给其它宝石;如果Ai<0,表示这颗宝石的能量过低,需要从其它宝石处获取-Ai的能量。保证∑Ai =0。只有当所有宝石的能量均相同时,把四叶草魔杖插入圆盘中央,才能开启超自然之界的通道。
不过,只有M对宝石之间可以互相传递能量,其中第i对宝石之间无论传递多少能量,都要花费Ti的代价。探险队员们想知道,最少需要花费多少代价才能使所有宝石的能量都相同?
输入格式
第一行两个整数N、M。
第二行N个整数Ai。
接下来M行每行三个整数pi,qi,Ti,表示在编号为pi和qi的宝石之间传递能量需要花费Ti的代价。数据保证每对pi、qi最多出现一次。
输出格式
输出一个整数表示答案。无解输出Impossible
样例输入
3 3
50 -20 -30
0 1 10
1 2 20
0 2 100
样例输出
30
提示
对于 50% 的数据,2<=N<=8。
对于 100% 的数据,2<=N<=16,0<=M<=N*(N-1)/2,0<=pi,qi<N,-1000<=Ai<=1000,0<=Ti<=1000,∑Ai=0。
做法一:最小生成树+状压dp
枚举点集,若当前点集构成联通块且能量之和为0,显然当前联通块传递能量的最小代价是其最小生成树,因此将每个这样的联通块看成一个物品,背包dp算出最小费用即可。
代码:
1 | #include<stdio.h> |
做法二:网络流
建图比较简单,从源点向每个$A[i]>0$的点连边,容量为$A[i]$,费用为0,从每个$A[i]<0$的点向汇点连边,容量为$-A[i]$,费用为0,然后点之间的边容量设为无穷。需要注意的是与普通的费用流不同,每次找到一条路时,要将有流经过的边费用改成0,将没有流经过的边费用改成原值,双向边都要改,因为第二次走是不产生费用的。最后再将有流的边的费用加起来。
代码:
1 |
|