某王   老师今天考了一套三国题,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. Bitmap转ImageSource

    bitmap to bytes Bitmap b = new Bitmap( "test.bmp "); MemoryStream ms = new MemoryStream(); ...

  2. python小白入门基础(三:整型)

    # Number(int float str complex) #int 整型(正整数 0 负整数)intvar_1 = 100print(intvar_1)invar_2 = 0 print(inv ...

  3. java的方法详解和总结

    一.什么是方法 在日常生活中,我们所说的方法就是为了解决某件事情,而采取的解决办法 java中的方法可以理解为语句的集合,用来完成解决某件事情或实现某个功能的办法 方法的优点: 程序变得更加简短而清晰 ...

  4. P1082 同余方程(拓展欧几里德)

    题目描述 求关于xx的同余方程 a x \equiv 1 \pmod {b}ax≡1(modb) 的最小正整数解. 输入输出格式 输入格式: 一行,包含两个正整数 a,ba,b,用一个空格隔开. 输出 ...

  5. 数据库行转列的sql语句

    问题描述 假设有张学生成绩表(CJ)如下Name Subject Result张三 语文 80张三 数学 90张三 物理 85李四 语文 85李四 数学 92李四 物理 82 现在 想写 sql 语句 ...

  6. 一文读懂神经网络训练中的Batch Size,Epoch,Iteration

    一文读懂神经网络训练中的Batch Size,Epoch,Iteration 作为在各种神经网络训练时都无法避免的几个名词,本文将全面解析他们的含义和关系. 1. Batch Size 释义:批大小, ...

  7. .NET Core加解密实战系列之——对称加密算法

    简介 加解密现状,编写此系列文章的背景: 需要考虑系统环境兼容性问题(Linux.Windows) 语言互通问题(如C#.Java等)(加解密本质上没有语言之分,所以原则上不存在互通性问题) 网上资料 ...

  8. windows-android-appium环境搭建

    一.安装jdk 安装jdk1.7以上版本,会生成一个jdk目录,和单独的jre目录(注意:不是jdk里面的jre,时安装过程中设置的那个jre路径)安装完成后并配置环境变量 在系统环境变量中,新建:J ...

  9. 基于Goc的Golang代码VSCode实时染色方案

    近日,Li Yiyang 老师基于Goc做了个VS Code插件,能够直观的看到被执行到的代码,当真充满想象力: 感兴趣的同学可以去goc仓库查看详情. Goc的核心能力就在于能够帮助我们在被测程序运 ...

  10. java 多线程-2

    七.线程生命周期 没错,线程也是有生命周期的.就好像人类有出生.儿童.青年.中年.晚年.死亡一般.下面是线程的生命周期图: 八.线程的安全问题 所谓线程不安全[并发问题],举个例子来说,如卖票,会出现 ...