[Luogu P3959] 宝藏 (状压DP+枚举子集)
题面
传送门:https://www.luogu.org/problemnew/show/P3959




Solution
这道题的是一道很巧妙的状压DP题。
首先,看到数据范围,应该状压DP没错了。
根据我们之前状压方程的设计经验,我们很快就能设计出这样的方程:
设f[i][j]表示用到第i个元素,当前连接状态为j的开销的min
但是我们很快就会发现,这个方程没法转移,因为随着连接方案的不同,新插入的点的K值会不同。
怎么办呢?
这时候我们可以重新设计一个巧妙的的状态。
重新阅读题目,我们可以发现题目中的K值可以理解为距离初始点的“层数”,下面这幅图可以简单的表示出来:

那么,我们可以考虑这样子设状态:
设f[i][j]表示到第i层,总共取了的点的状态为j。
这样的话,转移就可以取出来了:
f[i][j]=MIN(f[i-1][k]+trans[k][j]*(i-1)) (k为j的子集,即有可能转移到j的状态) (trans[k][j]表示从状态k转移到状态j的最小花费的路程)
trans需要暴力预处理出来。
怎么枚举子集呢?
如果2^n枚举就会T掉,因为我们枚举到了非子集的情况。
这里就引出了枚举子集的小技巧
对于状态x,它的子集为:p=x,p!=0,p=(p-1)&x (至于怎么证明,这里就不给出了,在草稿上推一推就会发现里面的精妙了)
答案就是min(f[i][2^n-1]),初始化f[1][2^(i-1)]=0 (i∈[1,n])
就酱,这道题就被我们切掉啦φ(>ω<*)
Code
//Luogu P3959 宝藏
//Sep,5th,2018
//状压DP+枚举子集小技巧
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
long long read()
{
long long x=0,f=1; char c=getchar();
while(!isdigit(c)){if(c=='-') f=-1;c=getchar();}
while(isdigit(c)){x=x*10+c-'0';c=getchar();}
return x*f;
}
const int N=12+2;
const int M=1<<N;
int n,m,dis[N][N],trans[M][M],POW[N];
long long f[N][M];
int main()
{
n=read(),m=read();
memset(dis,0x3f,sizeof dis);
for(int i=1;i<=m;i++)
{
int s=read(),t=read(),v=read();
if(dis[s][t]>v)
dis[s][t]=dis[t][s]=v;
} m=(1<<n);
POW[0]=1;
for(int i=1;i<=n;i++)
POW[i]=POW[i-1]*2;
for(int i=0;i<m;i++)
for(int j=i;j!=0;j=(j-1)&i)
{
bool OK=true;
int temp=i^j;
for(int k=n-1;k>=0;k--)
if(temp>=POW[k])
{
int tmin=0x3f3f3f3f;
for(int o=1;o<=n;o++)
if((POW[o-1]&j)==POW[o-1])
tmin=min(tmin,dis[o][k+1]);
if(tmin==0x3f3f3f3f)
{
OK=false;
break;
}
trans[j][i]+=tmin;
temp-=POW[k];
}
if(OK==false)
trans[j][i]=0x3f3f3f3f;
} /*cerr<<endl<<endl;
for(int i=0;i<m;i++)
for(int j=0;j<m;j++)
if(trans[i][j]!=0x3f3f3f3f and trans[i][j]!=0)
cerr<<i<<" "<<j<<" "<<trans[i][j]<<endl;*/ memset(f,0x3f,sizeof f);
for(int i=1;i<=n;i++)
f[1][POW[i-1]]=0;
for(int i=2;i<=n;i++)
for(int j=0;j<m;j++)
for(int k=j;k!=0;k=(k-1)&j)
if(trans[k][j]!=0x3f3f3f3f)
f[i][j]=min(f[i][j],f[i-1][k]+(i-1)*trans[k][j]); long long ans=0x3f3f3f3f3f3f3f3fll;
for(int i=1;i<=n;i++)
ans=min(ans,f[i][m-1]);
printf("%lld",ans);
return 0;
}
[Luogu P3959] 宝藏 (状压DP+枚举子集)的更多相关文章
- P3959 宝藏 状压dp
之前写了一份此题关于模拟退火的方法,现在来补充一下状压dp的方法. 其实直接在dfs中状压比较好想,而且实现也很简单,但是网上有人说这种方法是错的...并不知道哪错了,但是就不写了,找了一个正解. 正 ...
- 计蒜客习题:蒜头君的积木 (状压DP 枚举子集)
问题描述 蒜头君酷爱搭积木,他用积木搭了 n 辆重量为 wi的小车和一艘最大载重量为 W 的小船,他想用这艘小船将 n 辆小车运输过河.每次小船运载的小车重量不能超过 W.另外,小船在运载小车时,每辆 ...
- BZOJ 2560: 串珠子 (状压DP+枚举子集补集+容斥)
(Noip提高组及以下),有意者请联系Lydsy2012@163.com,仅限教师及家长用户. 2560: 串珠子 Time Limit: 10 Sec Memory Limit: 128 MB Su ...
- LOJ P3959 宝藏 状压dp noip
https://www.luogu.org/problemnew/show/P3959 考场上我怎么想不出来这么写的,状压白学了. 直接按层次存因为如果某个点在前面存过了则肯定结果更优所以不用在意各点 ...
- 【题解】P3959 宝藏 - 状压dp / dfs剪枝
P3959 宝藏 题目描述 参与考古挖掘的小明得到了一份藏宝图,藏宝图上标出了 n 个深埋在地下的宝藏屋, 也给出了这 n 个宝藏屋之间可供开发的m 条道路和它们的长度. 小明决心亲自前往挖掘所有宝 ...
- UVA 11825 Hackers’ Crackdown 状压DP枚举子集势
Hackers’ Crackdown Miracle Corporations has a number of system services running in a distributed com ...
- 洛谷P3959 宝藏(NOIP2017)(状压DP,子集DP)
洛谷题目传送门 Dalao的题解多数是什么模拟退火.DFS剪枝.\(O(3^nn^2)\)的状压DP之类.蒟蒻尝试着把状压改进了一下使复杂度降到\(O(3^nn)\). 考虑到每条边的贡献跟它所在的层 ...
- 洛谷$P3959\ [NOIp2017]$ 宝藏 状压$dp$
正解:状压$dp$ 解题报告: 传送门$QwQ$ $8102$年的时候就想搞这题了,,,$9102$了$gql$终于开始做这题了$kk$ 发现有意义的状态只有当前选的点集和深度,所以设$f_{i,j} ...
- [NOIP2017]宝藏 状压DP
[NOIP2017]宝藏 题目描述 参与考古挖掘的小明得到了一份藏宝图,藏宝图上标出了 n 个深埋在地下的宝藏屋, 也给出了这 n 个宝藏屋之间可供开发的 m 条道路和它们的长度. 小明决心亲自前往挖 ...
随机推荐
- python_购物车
流程图 实现方式 #!/usr/bin/python3 __author__ = 'beimenchuixue' __blog__ = 'http://www.cnblogs.com/2bjiuji ...
- Python numpy总结(3)——常用函数用法
1,np.ceil(x, y) 限制元素范围,进一法,即向上取整. x 表示输入的数据 y float类型 表示每个元素的上限. a = np.array([-1.7, -1.5, -0.2, 0. ...
- Optimisation
https://www.cnblogs.com/wuyudong/p/writing-efficient-c-and-code-optimization.html 1 不要过多使用 stack ,尽量 ...
- 011 01 Android 零基础入门 01 Java基础语法 02 Java常量与变量 05 变量的三个元素的详细介绍之三—— 变量值——即Java中的“字面值”
011 01 Android 零基础入门 01 Java基础语法 02 Java常量与变量 05 变量的三个元素的详细介绍之三-- 变量值--即Java中的"字面值" 变量值可以是 ...
- matlab中set设置图形属性
来源:https://ww2.mathworks.cn/help/matlab/ref/set.html?searchHighlight=set&s_tid=doc_srchtitle set ...
- C#与sql进行图片存取
1 using System; 2 using System.Collections.Generic; 3 using System.ComponentModel; 4 using System.Da ...
- 多测师讲解python_模块间的调用_高级讲师肖sir
案例1: 在aaa.py 文件A类中定义一个函数sadp: 在bbb.py文件中导入aaa模块,导入类 ,调用函数 案例2: aaa模块中定义一个A类, 在定义一个sadp的函数, 在bbb模块中导 ...
- 【API管理 APIM】APIM集成内部VNet后,自我访问出现(Unable to connect to the remote server)问题,而Remote Server正是APIM它自己
问题描述 在使用APIM配置内部VNET后,如API-1正常配置访问后端服务器的一个接口,而API-2则是通过调用APIM中的API-1来作为backendUrl,会出现500错误. 经过测试,目前这 ...
- Markdown基础知识
一 Markdown简介 Markdown是⼀种可以使⽤普通⽂本编辑器编写的标记语⾔,通过简单的标记语法,它可以使普通⽂本内容具有⼀定的格式,可以简单理解为纯⽂本格式的word. 软件⼀般⽤vscod ...
- postgresql使用规范解读
表设计规范1.建议能使用小字节数类型,就不要用大字节数类型2.建议能用varchar(N).text就不用char(N):3.建议使用default NULL,而不用default '':4.建议使用 ...