传送门:QAQQAQ

题意:

参与考古挖掘的小明得到了一份藏宝图,藏宝图上标出了$n$个深埋在地下的宝藏屋, 也给出了这$n$个宝藏屋之间可供开发的$m$条道路和它们的长度。

小明决心亲自前往挖掘所有宝藏屋中的宝藏。但是,每个宝藏屋距离地面都很远, 也就是说,从地面打通一条到某个宝藏屋的道路是很困难的,而开发宝藏屋之间的道路 则相对容易很多。

小明的决心感动了考古挖掘的赞助商,赞助商决定免费赞助他打通一条从地面到某 个宝藏屋的通道,通往哪个宝藏屋则由小明来决定。

在此基础上,小明还需要考虑如何开凿宝藏屋之间的道路。已经开凿出的道路可以 任意通行不消耗代价。每开凿出一条新道路,小明就会与考古队一起挖掘出由该条道路 所能到达的宝藏屋的宝藏。另外,小明不想开发无用道路,即两个已经被挖掘过的宝藏 屋之间的道路无需再开发。

新开发一条道路的代价是:

$L*K$

L代表这条道路的长度,K代表从赞助商帮你打通的宝藏屋到这条道路起点的宝藏屋所经过的 宝藏屋的数量(包括赞助商帮你打通的宝藏屋和这条道路起点的宝藏屋) 。

请你编写程序为小明选定由赞助商打通的宝藏屋和之后开凿的道路,使得工程总代 价最小,并输出这个最小值。

思路:看数据范围,$n<=12$,很自然而然想到状态压缩。

我们用$dp[i][mask]$保存从$i$开始挖,当前状态为$mask$的最少花费

现在的难点在于如何处理$K$——这个点在这棵树中的深度。

我们用$dis[i][mask]$保存在当前状态为$mask$,且当前答案最优,第$i$个点在树中的深度加一(为什么要加一?转移时好看一点嘛~~~)

我们可能会担心,当前$mask$最佳的连图方式无法保证它扩展到下一个节点的答案最优,但考虑到对于要被更新的$mask'$,总有一种之前最佳的$mask$连上一条边会使当前$mask'$答案最优。

而且被更新的$mask'$一定大于$mask$,所以每一个$dp[i][mask]$都会被更新到最优,那么我们在最后更新一下$ans$就可以啦~~~

代码:(之前并没有保存对于每一个$mask$的$dis$,导致可能最优连图和点的深度不对应,甚至答案会偏小(详见我洛谷上40分代码,即保存的是最小深度,而不是最优连图下的深度))

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=;
const ll inf=(int)2e9; ll dp[][],n,m;
ll E[][],dis[][]; void checkmin(ll &x,ll y)
{
if(x>y) x=y;
} int main()
{
scanf("%lld%lld",&n,&m);
for(int i=;i<n;i++)
{
for(int j=;j<n;j++) E[i][j]=inf;
}
int full=(<<n);
for(int i=;i<n;i++)
for(int j=;j<full;j++) dp[i][j]=inf;
for(int i=;i<=m;i++)
{
int x,y; ll z;
scanf("%d%d%lld",&x,&y,&z);
x--; y--; if(x==y) continue;
E[x][y]=min(z,E[x][y]);
E[y][x]=min(z,E[y][x]);
}
for(int start=;start<n;start++)
{
dis[(<<start)][start]=; dp[start][(<<start)]=;
for(int mask=;mask<full;mask++)
{
if(!(mask>>start)&) continue;
for(int from=;from<n;from++)
{
if(!(mask>>from)&) continue;
for(int to=;to<n;to++)
{
if((mask>>to)&) continue;
if(dp[start][mask]==inf||dis[mask][from]==) continue;
if(E[from][to]==inf) continue;
int mask1=mask|(<<to);
if(dp[start][mask1]>dp[start][mask]+dis[mask][from]*E[from][to])
{
dp[start][mask1]=dp[start][mask]+dis[mask][from]*E[from][to];
for(int p=;p<n;p++) dis[mask1][p]=dis[mask][p];
dis[mask1][to]=dis[mask][from]+;
}
}
}
}
}
ll ans=inf;
for(int i=;i<n;i++) checkmin(ans,dp[i][full-]);
cout<<ans<<endl;
return ;
}

(本来以为这种做法是错的,兴致勃勃地造了一组数据结果发现hack不掉。。。现在才真正懂)

洛谷P3959——宝藏的更多相关文章

  1. 洛谷P3959 宝藏(NOIP2017)(状压DP,子集DP)

    洛谷题目传送门 Dalao的题解多数是什么模拟退火.DFS剪枝.\(O(3^nn^2)\)的状压DP之类.蒟蒻尝试着把状压改进了一下使复杂度降到\(O(3^nn)\). 考虑到每条边的贡献跟它所在的层 ...

  2. 洛谷 P3959 宝藏 解题报告

    P3959 宝藏 题目描述 参与考古挖掘的小明得到了一份藏宝图,藏宝图上标出了 \(n\) 个深埋在地下的宝藏屋, 也给出了这 \(n\) 个宝藏屋之间可供开发的 \(m\) 条道路和它们的长度. 小 ...

  3. 洛谷P3959 宝藏

    去年NOIP第二毒瘤(并不)的题终于被我攻克了,接下来就只剩noip难度巅峰列队了. 首先说一下三种做法:随机化,状压DP和搜索. 前两种做法我都A了,搜索实在是毒瘤,写鬼啊. 有些带DFS的记忆化搜 ...

  4. 2018.08.09洛谷P3959 宝藏(随机化贪心)

    传送门 回想起了自己赛场上乱搜的20分. 好吧现在也就是写了一个随机化贪心就水过去了,不得不说随机化贪心大法好. 代码: #include<bits/stdc++.h> using nam ...

  5. 洛谷P3959 宝藏(状压dp)

    传送门 为什么感觉状压dp都好玄学……FlashHu大佬太强啦…… 设$f_{i,j}$表示当前选的点集为$i$,下一次要加入的点集为$j$时,新加入的点和原有的点之间的最小边权.具体的转移可以枚举$ ...

  6. 洛谷P3959 宝藏(模拟退火乱搞)

    题意 题目链接 题面好长啊...自己看吧.. Sol 自己想了一个退火的思路,没想到第一次交85,多退了几次就A了哈哈哈 首先把没用的边去掉,然后剩下的边从小到大排序 这样我们就得到了一个选边的序列, ...

  7. 洛谷 P3959 宝藏【状压dp】

    一开始状态设计错了-- 设f[i][s]为当前与根节点联通状况为s,最深深度为i 转移的话枚举当前没有和根联通的点集,预处理出把这些点加进联通块的代价(枚举s中的点和当前点的连边乘以i即可),然后用没 ...

  8. 【题解】洛谷P3959 [NOIP2017TG] 宝藏(状压DP+DFS)

    洛谷P3959:https://www.luogu.org/problemnew/show/P3959 前言 NOIP2017时还很弱(现在也很弱 看出来是DP 但是并不会状压DP 现在看来思路并不复 ...

  9. NOIP2017提高组Day2T2 宝藏 洛谷P3959 状压dp

    原文链接https://www.cnblogs.com/zhouzhendong/p/9261079.html 题目传送门 - 洛谷P3959 题目传送门 - Vijos P2032 题意 给定一个 ...

随机推荐

  1. iscroll refresh无效解决办法

    最近用iscroll.js 写移动页面,效果还是挺好的.但,还是会遇到重新初始化的问题. var myScroll = new IScroll('#rule_wrapper',{ click:true ...

  2. 数学思维——cf1244C

    可惜cf不能用int128,不然这个题就是个exgcd的板子题 这是exgcd的解法,但是只用ll的话会溢出 #include<bits/stdc++.h> using namespace ...

  3. HDU 6590 Code (判断凸包相交)

    2019 杭电多校 1 1013 题目链接:HDU 6590 比赛链接:2019 Multi-University Training Contest 1 Problem Description Aft ...

  4. java读取pdf文本转换html

    补充:一下代码基于maven,现将依赖的jar包单独导出 地址:pdf jar 完整代码地址 也就两个文件 java读取pdf中的纯文字,这里使用的是pdfbox工具包 maven引入如下配置 < ...

  5. 调用API接口,查询手机号码归属地(2)

    使用pymysql pip install pymysql 创建mysql测试表 CREATE TABLE `userinfo` ( `id` int(20) NOT NULL AUTO_INCREM ...

  6. 44-Ubuntu-用户管理-09-chmod的数字表示法介绍

    chmod 修改文件和目录权限 chmod在设置权限时,可以简单地使用三个数字分别对应拥有者/组和其他用户的权限. 注意: chmod直接修改文件|目录的'读|写|执行'权限,但是不能精确到拥有者|组 ...

  7. poj 1905 图形推算+二分

    参考博客: 题意: 一根两端固定在两面墙上的杆 受热弯曲后变弯曲 求前后两个状态的杆的中点位置的距离 分析:见博客 代码: #include<stdio.h> #include<io ...

  8. swagger2 注解说明 ( @ApiImplicitParams )

    @Api:用在请求的类上,表示对类的说明 tags="说明该类的作用,可以在UI界面上看到的注解" value="该参数没什么意义,在UI界面上也看到,所以不需要配置&q ...

  9. 【转】console.dir()和console.log()的区别

    原文链接:https://blog.csdn.net/ky1in93/article/details/80828499 console对象详解:https://segmentfault.com/a/1 ...

  10. Java 整型数组只有一个整数是只出现一次的,其余都出现两次

    package writtenExamination; /** * @author lindaZ * @date 2015年10月22日 * @todo 找出数组中仅出现一次的数,其余数字出现两次 * ...