BZOJ_3058_四叶草魔杖_kruscal+状压DP

BZOJ_3058_四叶草魔杖_kruscal+状压DP

Description

魔杖护法Freda融合了四件武器,于是魔杖顶端缓缓地生出了一棵四叶草,四片叶子幻发着淡淡的七色光。圣剑护法rainbow取出了一个圆盘,圆盘上镶嵌着N颗宝石,编号为0~N-1。第i颗宝石的能量是Ai。如果Ai>0,表示这颗宝石能量过高,需要把Ai的能量传给其它宝石;如果Ai<0,表示这颗宝石的能量过低,需要从其它宝石处获取-Ai的能量。保证∑Ai =0。只有当所有宝石的能量均相同时,把四叶草魔杖插入圆盘中央,才能开启超自然之界的通道。
不过,只有M对宝石之间可以互相传递能量,其中第i对宝石之间无论传递多少能量,都要花费Ti的代价。探险队员们想知道,最少需要花费多少代价才能使所有宝石的能量都相同?

Input

第一行两个整数N、M。
第二行N个整数Ai。
接下来M行每行三个整数pi,qi,Ti,表示在编号为pi和qi的宝石之间传递能量需要花费Ti的代价。数据保证每对pi、qi最多出现一次。

Output

输出一个整数表示答案。无解输出Impossible。

Sample Input

3 3
50 -20 -30
0 1 10
1 2 20
0 2 100

Sample Output

30

HINT

对于 100% 的数据,2<=N<=16,0<=M<=N*(N-1)/2,0<=pi,qi<N,-1000<=Ai<=1000,0<=Ti<=1000,∑Ai=0。

 


 

可以转化成这样一个问题:将原图划分成若干个连通块,使得每个连通块内权值和为0,求最小代价。

由于n特别小可以状压,f[i]表示状态为i时最小的联通代价。

对于每个和为0的集合,用kruscal求使这个集合连通的最小代价,然后DP即可。

 

代码:

 

#include <cstdio>
#include <string.h>
#include <algorithm>
using namespace std;
int cal[1<<16],n,m,a[20],f[1<<16],g[1<<16],fa[20];
struct E {
	int x,y,z;
	bool operator < (const E &u) const {
		return z<u.z;
	}
}e[350];
int find(int x) {return fa[x]==x?x:fa[x]=find(fa[x]);}
void dfs(int dep,int sta,int sum) {
	if(dep==n) {
		cal[sta]=sum; return ;
	}
	dfs(dep+1,sta|(1<<dep),sum+a[dep+1]);
	dfs(dep+1,sta,sum);
}
void build(int sta) {
	int i,num=0;
	for(i=1;i<=n;i++) {
		fa[i]=i; if(sta&(1<<(i-1))) num++;
	}
	int re=0,cnt=0;
	for(i=1;i<=m;i++) {
		int dx=find(e[i].x),dy=find(e[i].y);
		if((sta&(1<<(dx-1)))&&(sta&(1<<(dy-1)))&&dx!=dy) {
			fa[dx]=dy; re+=e[i].z; cnt++; if(cnt==num-1) break;
		}
	}
	if(cnt==num-1) f[sta]=re;
}
int main() {
	// freopen("shield.in","r",stdin);
	// freopen("shield.out","w",stdout);
	scanf("%d%d",&n,&m);
	int i,j;
	for(i=1;i<=n;i++) scanf("%d",&a[i]);
	dfs(0,0,0);
	for(i=1;i<=m;i++) {
		scanf("%d%d%d",&e[i].x,&e[i].y,&e[i].z); e[i].x++; e[i].y++;
	}
	sort(e+1,e+m+1);
	int mask=(1<<n)-1; 
	memset(f,0x3f,sizeof(f));
	f[0]=0;
	for(i=1;i<=mask;i++) {
		if(cal[i]==0) {
			build(i);
		}
	}
	for(i=0;i<=mask;i++) {
		for(j=i&(i-1);j;j=i&(j-1)) {
			if(f[j]<=1000000&&f[i-j]<=1000000) f[i]=min(f[i],f[j]+f[i-j]);
		}
	}
	if(f[mask]<=10000000) printf("%d\n",f[mask]);
	else puts("Impossible");
}
/*
3 3
50 -20 -30
0 1 10
1 2 20
0 2 100
*/

 

posted @ 2018-06-07 07:21  fcwww  阅读(310)  评论(0编辑  收藏  举报