某王   老师今天考了一套三国题,AK了。。。就挑一道最恶心的题来写一写吧。

题目描述:

【题目背景】

公元215年,刘备取益州,孙权令诸葛瑾找刘备索要荆州。刘备不答应,孙权极为恼恨,便派吕蒙率军取长沙、零陵、桂阳三郡。长沙、桂阳蜀将当即投降。刘备得知后,亲自从成都赶到公安(今湖北公安),派大将关羽争夺三郡。孙权也随即进驻陆口,派鲁肃屯兵益阳,抵挡关羽。双方剑拔弩张,孙刘联盟面临破裂,在这紧要关头,鲁肃为了维护孙刘联盟,不给曹操可乘之机,决定当面和关羽商谈。“肃邀羽相见,各驻兵马百步上,但诸将军单刀俱会”。双方经过会谈,缓和了紧张局势。随后,孙权与刘备商定平分荆州,“割湘水为界,于是罢军”,孙刘联盟因此能继续维持。

【问题描述】

关羽受鲁肃邀请,为了大局,他决定冒险赴会。他带着侍从周仓,义子关平,骑着赤兔马,手持青龙偃月刀,从军营出发了,这就是历史上赫赫有名的“单刀赴会”。关羽平时因为军务繁重,决定在这次出行中拜访几个多日不见的好朋友。然而局势紧张,这次出行要在限定时间内完成,关公希望你能够帮助他安排一下行程,安排一种出行方式,使得从军营出发,到达鲁肃处赴会再回来,同时拜访到尽可能多的朋友,在满足这些条件下行程最短。注意拜访朋友可以在赴会之前,也可以在赴会之后。现在给出地图,请你完成接下来的任务。

输入

第一行n,m,k,t,代表有n个地点,m条道路,有k个朋友(不包括鲁肃),以及限定时间t(行走1单位长度的路程用时1单位时间)。

接下来m行,每行有x,y,w三个整数,代表x和y之间有长度为w的道路相连。

接下来一行有k个整数,代表朋友所在的都城编号(保证两两不同,且不在1和n)

(我们约定1是关羽的营地,n是鲁肃的营地)

输出

输出两个整数,分别是最多可以拜访的朋友数,以及在这种情况下最少需要耗费的时间,如果连到达鲁肃再回来都无法完成,输出一个-1就可以了。

样例输入

5 7 2 15
1 2 5
1 3 3
2 3 1
2 4 1
3 4 4
2 5 2
4 5 3
2 4

样例输出

2 14

提示

【数据规模和约定】

有10%数据,n<=10,m<=50,k<=5;

有10%数据,k=0;

有10%数据,k=1;

另30%数据,k<=5;

对于100%数据,n<=10000,m<=50000,k<=15,t<=2147483647,w<=10000

思路分析:

很清楚的数据范围,很好想的状压DP,很恶心的代码实现。

嗯,这就是我对这题的评价了。

读完题目,很清楚,整张图中只有k+2个点是有用的,分别为:关羽的营地、k个朋友的家以及鲁肃的营地。所以我们可以对其进行预处理,跑k+2遍单源最短路(我跑的是Dijkstra),然后就可以开始状压了!

既然是状压DP,那么我们应该压啥呢?——还能压啥啊?压k走起啊!

二进制状态sta,每一位上的0/1表示这个朋友也没有被访问过。那么我们就可以顺利地写出状态:dp[sta,i]表示在状态sta下,关羽现在正在第i位朋友的家中,所花费的时间的最小值。

那么我们便可以通过再枚举一个j表示关羽下一次要去第j个朋友的家,进行转移。转移方程便为:

dp[sta|1<<j-1,j]=min(dp[sta|1<<j-1,j],dp[sta,i]+d[i,a[j]]);

其中d数组,d[i,j]表示以i为源点到城市j的最小时间花费。a数组,a[i]表示关羽的第i位朋友住在城市a[i]。

时间复杂度就是n2*2n+k*(n+m)log n。

那么有些人可能会想:能不能用类似于【愤怒的小鸟】的优化方法把状压DP的时间优化成n*2n呢?

深入思考一下发现这其实是不可行的。

我们能把【愤怒的小鸟】优化掉一个n是因为每一只猪都必须被打掉,而且先打和后打是没有区别的,所以我们可以强制让他打一只猪,而这题不一样,关羽并不是一定要访问完k个朋友,没有他一定要访问的朋友,所以不能采用类似的方法来优化。

代码实现:

type
hehe=record
dist,id:longint;
end;
var
f:array[1..2000000]of hehe;
s:array[1..150000]of longint;
dp:array[0..150000,0..20]of longint;
a:array[0..20]of longint;
head,visit:array[1..10000]of longint;
d:array[1..20,1..10000]of longint;
next,vet,dist:array[1..100000]of longint;
n,m,t,i,j,k,sta,weight,x,y,z,tot,oo,ans,mi:longint;
procedure add(x,y,z:longint);
begin
inc(tot);
next[tot]:=head[x];
head[x]:=tot;
vet[tot]:=y;
dist[tot]:=z;
end;
function min(x,y:longint):longint;
begin
if x<y then exit(x) else exit(y);
end;
procedure swap(x,y:longint);
var
t:hehe;
begin
t:=f[x]; f[x]:=f[y]; f[y]:=t;
end;
procedure doit(x,k:longint);
begin
if x>1<<17 then exit;
s[x]:=k;
doit(x*2,k); doit(x*2+1,k+1);
end;
procedure up(x:longint);
begin
if x=1 then exit;
if f[x].dist<f[x div 2].dist then begin swap(x,x div 2); up(x div 2); end;
end;
procedure down(x:longint);
var
k:longint;
begin
if x*2>weight then exit;
k:=min(f[x*2].dist,f[x*2+1].dist);
if f[x].dist>k then
if k=f[x*2].dist then begin swap(x,x*2); down(x*2); end
else begin swap(x,x*2+1); down(x*2+1); end;
end;
procedure push(x,id:longint);
begin
inc(weight);
f[weight].dist:=x;
f[weight].id:=id;
up(weight);
end;
procedure pop;
begin
x:=f[1].id;
f[1].dist:=f[weight].dist;
f[1].id:=f[weight].id;
f[weight].dist:=oo;
f[weight].id:=oo;
dec(weight);
down(1);
end;
procedure dijkstra(k,s:longint);
var
y,i:longint;
begin
fillchar(f,sizeof(f),$7f);
fillchar(visit,sizeof(visit),0);
push(0,s); d[k,s]:=0;
while weight>0 do
begin
pop;
if visit[x]=1 then continue;
i:=head[x]; visit[x]:=1;
while i<>0 do
begin
y:=vet[i];
if d[k,y]>d[k,x]+dist[i] then
begin d[k,y]:=d[k,x]+dist[i]; push(d[k,y],y); end;
i:=next[i];
end;
end;
end;
begin
doit(1,1);
read(n,m,k,t);
for i:=1 to m do
begin
read(x,y,z);
add(x,y,z); add(y,x,z);
end;
fillchar(d,sizeof(d),$7f); oo:=d[1,1];
dijkstra(1,1); inc(k); a[1]:=1;
for i:=2 to k do
begin
read(a[i]);
dijkstra(i,a[i]);
end;
inc(k); a[k]:=n;
dijkstra(k,n);
fillchar(dp,sizeof(dp),$7f);
dp[1,1]:=0;
for sta:=1 to 1<<k-2 do
for i:=1 to k do
if dp[sta,i]<>oo then
for j:=2 to k do
dp[(sta)or(1<<(j-1)),j]:=min(dp[(sta)or(1<<(j-1)),j],dp[sta,i]+d[i,a[j]]);
mi:=oo;
for sta:=1 to 1<<k-1 do
if sta>>(k-1)=1 then
for i:=2 to k do
if dp[sta,i]+d[i,1]<=t then
if s[sta]-2>ans then begin ans:=s[sta]-2; mi:=dp[sta,i]+d[i,1]; end
else
if (s[sta]-2=ans)and(dp[sta,i]+d[i,1]<mi) then mi:=dp[sta,i]+d[i,1];
if mi=oo then writeln(-1) else writeln(ans,' ',mi);
end.

状压DP——【蜀传之单刀赴会】的更多相关文章

  1. [poj1185]炮兵阵地_状压dp

    炮兵阵地 poj-1185 题目大意:给出n列m行,在其中添加炮兵,问最多能加的炮兵数. 注释:n<=100,m<=10.然后只能在平原的地方建立炮兵. 想法:第2到状压dp,++.这题显 ...

  2. [转]状态压缩dp(状压dp)

    状态压缩动态规划(简称状压dp)是另一类非常典型的动态规划,通常使用在NP问题的小规模求解中,虽然是指数级别的复杂度,但速度比搜索快,其思想非常值得借鉴. 为了更好的理解状压dp,首先介绍位运算相关的 ...

  3. nyoj1273 河南省第九届省赛_"宣传墙"、状压DP+矩阵幂加速

    宣传墙 时间限制:1000 ms  |  内存限制:65535 KB 难度:4 描述 ALPHA 小镇风景美丽,道路整齐,干净,到此旅游的游客特别多.CBA 镇长准备在一条道路南 面 4*N 的墙上做 ...

  4. 最小总代价 状压DP

    描述 n个人在做传递物品的游戏,编号为1-n. 游戏规则是这样的:开始时物品可以在任意一人手上,他可把物品传递给其他人中的任意一位:下一个人可以传递给未接过物品的任意一人. 即物品只能经过同一个人一次 ...

  5. BZOJ_3058_四叶草魔杖_kruscal+状压DP

    BZOJ_3058_四叶草魔杖_kruscal+状压DP Description 魔杖护法Freda融合了四件武器,于是魔杖顶端缓缓地生出了一棵四叶草,四片叶子幻发着淡淡的七色光.圣剑护法rainbo ...

  6. [tyvj2054] 四叶草魔杖 (最小生成树 状压dp)

    传送门 Background 陶醉在彩虹光芒笼罩的美景之中,探险队员们不知不觉已经穿过了七色虹,到达了目的地,面前出现了一座城堡和小溪田园,城堡前的木牌上写着"Poetic Island&q ...

  7. BZOJ 1087: [SCOI2005]互不侵犯King [状压DP]

    1087: [SCOI2005]互不侵犯King Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3336  Solved: 1936[Submit][ ...

  8. nefu1109 游戏争霸赛(状压dp)

    题目链接:http://acm.nefu.edu.cn/JudgeOnline/problemShow.php?problem_id=1109 //我们校赛的一个题,状压dp,还在的人用1表示,被淘汰 ...

  9. poj3311 TSP经典状压dp(Traveling Saleman Problem)

    题目链接:http://poj.org/problem?id=3311 题意:一个人到一些地方送披萨,要求找到一条路径能够遍历每一个城市后返回出发点,并且路径距离最短.最后输出最短距离即可.注意:每一 ...

随机推荐

  1. Spine学习六 - 碰撞检测

    相信在使用Spine做游戏的时候,肯定会遇到这样的需求: 一个人物有一把大刀,要使用这把大刀去砍怪,伤害检测以这把大刀砍刀怪物为准,那么要怎么在一个看上去就是一体的Spine Object上绑定一个碰 ...

  2. Codeforces 1389 题解(A-E)

    AC代码 A. LCM Problem 若\(a < b\),则\(LCM(a,b)\)是\(a\)的整数倍且\(LCM(a,b) \ne a\),所以\(LCM(a,b) \ge 2a\),当 ...

  3. [BUUOJ记录] [GYCTF]EasyThinking

    主要考察ThinkPHP6.0的一个任意文件写入的CVE以及突破disable_function的方法. ThinkPHP6.0.0任意文件操作漏洞 理论分析 进入题目是一个简单的操作页面,dirma ...

  4. java中变量在内存的位置

    package day02; /* * 成员变量:在堆内存中,因为对象的存在,才在内存中存在:作用于整改类中 * 局部变量:在栈内存中:作用于函数中,或者语句中 * */ class car{ //描 ...

  5. Agumaster 增加股票表台账页面

  6. 你会Spring Cloud吗?

    如果有人问你这句话,你不要急着回答,因为答语和问者的关系莫大. 如果问你的是个老板或管理者,那么他多半不了解这项技术,很有可能只是知道这个技术名词而已,你知道的一定比他们深,你大可夸夸其谈的.在口若悬 ...

  7. JavaScript浮点数及其运算

    .普及两个函数Math.pow(底数,几次方)Number.toFixed(小数位数)2.浮点数相加function accAdd(arg1,arg2){   var r1,r2,m;   try{r ...

  8. WebSphere MQ常用命令及配置

    WebSphere MQ常用命令及配置 (2012-06-23 23:09:16) 标签: mq命令 杂谈 分类: MQ [导读]WebSphere MQ常用命令及配置 一,队列管理命令 1,创建队列 ...

  9. ASP.NET Core新书终于上市,完成今年一个目标,赠书活动

    2018年.NET Core 2.0发布后,开始逐步学习.NET Core 并逐步在新的项目中使用ASP.NET Core.并且零零散散写的写了将近30篇学习笔记发到园子里,包括ASP.NET Cor ...

  10. 如何解决Python下 pip install module 下载慢解决方法?

    对于Python来编程的用户最大的一个痛点就是,下载模块是下载速度特别慢,那么有没有解决方法呢? 换Python的pip下载源 1.首先安装一个模块 pqi,在cmd下 pip install pqi ...