JZOJ3392-四叶草魔杖

Description&Data Constraint

魔杖护法Freda融合了四件武器,于是魔杖顶端缓缓地生出了一棵四叶草,四片叶子焕发着淡淡的七色光。圣剑护法rainbow取出了一个圆盘,圆盘上镶嵌着N颗宝石,编号为0~N-1。第i颗宝石的能量是Ai。如果Ai>0,表示这颗宝石能量过高,需要把Ai的能量传给其他宝石;如果Ai<0,表示这颗宝石的能量过低,需要从其他宝石处获取-Ai的能量。保证 ∑ A i = 0 \sum A_i=0 Ai=0。只有当所有宝石的能量均相同时,把四叶草魔杖插入圆盘中央,才能开启超自然之界的通道。

不过,只有M对宝石之间可以互相传递能量,其中第i对宝石之间无论传递多少能量,都要花费Ti的代价。探险队员们想知道,最少需要花费多少代价才能使所有宝石的能量都相同?

2 ≤ N ≤ 16 , 0 ≤ M ≤ N × ( N − 1 ) 2 , 0 ≤ p i , q i ≤ N , − 1 0 3 ≤ A i ≤ 1 0 3 , 0 ≤ T i ≤ 1 0 3 , ∑ A i = 0 2\le N\le16,0\le M \le \dfrac{N\times(N-1)}{2},0\le p_i,q_i\le N,-10^3\le A_i\le 10^3,0\le T_i\le 10^3,\sum A_i=0 2N16,0M2N×(N1),0pi,qiN,103Ai103,0Ti103,Ai=0

Solution

由于 N N N很小,可以在 O ( N × 2 N ) O(N\times2^N) O(N×2N)的时间内把所有联通的且点权和为0的集合找出来,每个集合跑一遍最小生成树

然后考虑状压DP,设 f s f_s fs表示 s s s的点权和为0的最小代价,那么部分初值就是找出来的为0集合的最小生成树

然后暴力枚举两个子集 i i i j j j,如果 i i i j j j没有交集,那么 f i ∣ j = f i + f j f_{i|j}=f_i+f_j fij=fi+fj

Code

#include<cstdio>
#include<cstring>
#include<algorithm>
#define N 20
#define inf 0X3f3f3f3f
using namespace std;
struct node
{
	int x,y,v;
}edge[N*N];
int n,m,x,y,z,xx,yy,ans,res,tot,a[N],f[N],sum[1<<N][N],MAP[N][N],RES[N],dp[1<<N];
bool b[N],bz[N],bj[N];
bool cmp(node x,node y) {return x.v<y.v;}
int find(int x)
{
	if (f[x]!=x) f[x]=find(f[x]);
	return f[x];
}
void kruskal(int x)
{
	int num=0;
	for (int i=1;i<=sum[x][0];++i)
		for (int j=i+1;j<=sum[x][0];++j)
		{
			if (MAP[sum[x][i]][sum[x][j]])
			{
				edge[++num].x=sum[x][i];
				edge[num].y=sum[x][j];
				edge[num].v=MAP[sum[x][i]][sum[x][j]];	
			}	
		}
	for (int i=1;i<=n;++i) f[i]=i;	
	sort(edge+1,edge+num+1,cmp);
	memset(b,true,sizeof(b));
	RES[x]=0;
	for (int i=1;i<=num;++i)
	{
		int X=edge[i].x,Y=edge[i].y,Z=edge[i].v;
		if (!b[X]||!b[Y]) continue;
		xx=find(X);yy=find(Y);
		if (xx!=yy)
		{
			f[xx]=yy;
			RES[x]+=Z;
		}
	}
}
void dfs(int x)
{
	++res;
	bz[x]=true;
	for (int i=1;i<=n;++i)
	{
		if (!bj[i]) continue;
		if (bz[i]) continue;
		if (MAP[x][i]) dfs(i);
	}
}
void dg(int x,int Sum,bool flag)
{
	if (x>n)
	{
		if (!flag) return;
		if (Sum==0)
		{
			res=0;
			int pd0=0,pd[N];
			memset(bz,false,sizeof(bz));
			memset(pd,0,sizeof(pd));
			for (int i=1;i<=n;++i)
				if (bj[i]) pd[++pd0]=i;
			dfs(pd[1]);
			if (res<pd0) return;
			++tot;
			for (int i=1;i<=n;++i)
			{
				if (bj[i])
				{
					++sum[tot][0];
					sum[tot][sum[tot][0]]=i;
				}
			}
			kruskal(tot);
		}
		return;
	}
	bj[x]=true;
	dg(x+1,Sum+a[x],true);
	bj[x]=false;
	dg(x+1,Sum,flag);
}
int main()
{
	scanf("%d%d",&n,&m);
	for (int i=1;i<=n;++i)
		scanf("%d",&a[i]);
	for (int i=1;i<=m;++i)
	{
		scanf("%d%d%d",&x,&y,&z);
		++x;++y;
		MAP[x][y]=MAP[y][x]=z;
	}
	dg(1,0,false);
	memset(dp,inf,sizeof(dp));
	for (int i=1;i<=tot;++i)
	{
		int ss=0;
		for (int j=1;j<=sum[i][0];++j)
			ss+=(1<<(sum[i][j]-1));
		dp[ss]=RES[i];
	}
	for (int i=0;i<(1<<n);++i)
	{
		if (dp[i]==inf) continue;
		for (int j=0;j<(1<<n);++j)
		{
			if (dp[j]==inf) continue;
			if ((i&j)==0) dp[i|j]=min(dp[i|j],dp[i]+dp[j]);
		}
	}
	if (dp[(1<<n)-1]==inf) printf("Impossible\n");
	else printf("%d\n",dp[(1<<n)-1]);
	return 0;
} 
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值