最佳挑水

描述:

  小Y住在农村,离他的家不远有一口井,传说是小Y的祖先开掘的。虽然小Y的村子里通了自来水,但由于这口井的井水质量非常好,因此小Y家仍然喝这口井里的水。小Y非常喜欢这口井,所以他经常去挑水。

  小Y的家里有n(n是偶数)只桶,这些桶虽然大小相等,但是由于很多都有些破损,所以认为它们是不同的。小Y经常挑一根扁担(当然一定是带两只空桶)去井边挑水。小Y每次去井旁都会把桶中的水装到极限(假设水量无穷,且小Y都能够挑得动)。设小Y挑得是i,j两只桶,则挑水一趟需要走time[i,j]分钟。小Y想要在最少的时间内用自己的力量把家里所有的空桶装满。

  小Y觉得这是个难题,于是来找你帮忙编写一个程序来找出一种最佳挑水方案。


输入:

  输入文件中的的第一行为一个整数n(4<=n<=18)。

接下来的n行,每行有n个数,表示了time矩阵。

其中:time矩阵中每一个数都是小于等于32768正整数,且time[i,i]是没有用的。

注意:time[i,j]=time[j,i]。

输出:
  输出文件中仅一行为一个数,即最佳挑水方案的最少时间。


样例输入:
4
0 100 5 100
100 0 100 11
5 100 0 100
100 11 100 0

样例输出:

16


题解:

太多的状压dp都是被数据范围出卖的(笑);

20的范围来压缩已经取了哪些桶;

f[i]表示这个状态下的最小时间值;

转移就把上一个的去掉两个桶然后加上time[i][j];


状压太不好转移了。。。

我觉得宽搜还是比较容易想到并且理解的= =;


代码:

#include<queue>
#include<stdio.h>
#include<string.h>
#include<algorithm>
#define N 20
using namespace std;
int n,map[20][20],f[1000000];
bool vis[1000000];
queue<int>q;
int main()
{
	int m,i,j,k,now,t1,t2,y;
	scanf("%d",&n);
	for(i=1,k=0;i<=n;i++)
	{
		map[i][i]=0x3f3f3f3f;
		for(j=1;j<=n;j++)
		{
			scanf("%d",&map[i][j]);
		}
		k=(k|1)<<1;
	}
	memset(f,0x3f,sizeof(f));
	f[0]=0;
	vis[0]=1;
	q.push(0);
	while((now=q.front())!=k)
	{
		q.pop();
		i=1,t1=now;
		while(i<n)
		{
			while(t1&2)
			{
				t1>>=1;
				i++;
			}
			t2=t1>>1;
			for(j=i+1;j<=n;j++)
			{
				if((t2&2)==0)
				{
					f[y=(now|(1<<i)|(1<<j))]=min(f[y],f[now]+map[i][j]);
					if(vis[y]==0)
					{
						vis[y]=1;
						q.push(y);
					}
				}
				t2>>=1;
			}
			t1>>=1;
			i++;
		}
	}
	printf("%d",f[k]);
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值