【BZOJ1097】[POI2007]旅游景点atr 最短路+状压DP

【BZOJ1097】[POI2007]旅游景点atr

Description

  FGD想从成都去上海旅游。在旅途中他希望经过一些城市并在那里欣赏风景,品尝风味小吃或者做其他的有趣的事情。经过这些城市的顺序不是完全随意的,比如说FGD不希望在刚吃过一顿大餐之后立刻去下一个城市登山,而是希望去另外什么地方喝下午茶。幸运的是,FGD的旅程不是既定的,他可以在某些旅行方案之间进行选择。由于FGD非常讨厌乘车的颠簸,他希望在满足他的要求的情况下,旅行的距离尽量短,这样他就有足够的精力来欣赏风景或者是泡MM了^_^.整个城市交通网络包含N个城市以及城市与城市之间的双向道路M条。城市自1至N依次编号,道路亦然。没有从某个城市直接到它自己的道路,两个城市之间最多只有一条道路直接相连,但可以有多条连接两个城市的路径。任意两条道路如果相遇,则相遇点也必然是这N个城市之一,在中途,由于修建了立交桥和下穿隧道,道路是不会相交的。每条道路都有一个固定长度。在中途,FGD想要经过K(K<=N-2)个城市。成都编号为1,上海编号为N,而FGD想要经过的N个城市编号依次为2,3,…,K+1.举例来说,假设交通网络如下图。FGD想要经过城市2,3,4,5,并且在2停留的时候在3之前,而在4,5停留的时候在3之后。那么最短的旅行方案是1-2-4-3-4-5-8,总长度为19。注意FGD为了从城市2到城市4可以路过城市3,但不在城市3停留。这样就不违反FGD的要求了。并且由于FGD想要走最短的路径,因此这个方案正是FGD需要的。

Input

  第一行包含3个整数N(2<=N<=20000),M(1<=M<=200000),K(0<=K<=20),意义如上所述。

Output

  只包含一行,包含一个整数,表示最短的旅行距离。

Sample Input

8 15 4
1 2 3
1 3 4
1 4 4
1 6 2
1 7 3
2 3 6
2 4 2
2 5 2
3 4 3
3 6 3
3 8 6
4 5 2
4 8 6
5 7 4
5 8 6
3
2 3
3 4
3 5

Sample Output

19

HINT

 上面对应于题目中给出的例子。

题解:先预处理出每个关键点到其他点的最短路,然后跑状压DP即可。对于关键点的先后顺序,可以先处理出每个关键点需要的前置节点,那么只有当前状态包含所有前置节点才能转移。

#include <cstdio>
#include <iostream>
#include <cstring>
#include <queue>
#include <utility>
#define mp(A,B) make_pair(A,B)
using namespace std;
typedef pair<int,int> pii;
int n,m,k,kk,cnt,ans;
int to[400010],next[400010],val[400010],head[20010],r[21],rr[21];
int dis[20010],v[25][25],vis[20010],f[(1<<20)+10][21],g[20010];
priority_queue<pii> q;
inline int rd()
{
	int ret=0,f=1;	char gc=getchar();
	while(gc<‘0‘||gc>‘9‘)	{if(gc==‘-‘)f=-f;	gc=getchar();}
	while(gc>=‘0‘&&gc<=‘9‘)	ret=ret*10+gc-‘0‘,gc=getchar();
	return ret*f;
}
void add(int a,int b,int c)
{
	to[++cnt]=b,val[cnt]=c,next[cnt]=head[a],head[a]=cnt;
}
void dijkstra(int S)
{
	memset(dis,0x3f,sizeof(dis));
	memset(vis,0,sizeof(vis));
	while(!q.empty())	q.pop();
	q.push(mp(0,S)),dis[S]=0;
	int i,u,tmp=0;
	while(!q.empty())
	{
		u=q.top().second,q.pop();
		if(vis[u])	continue;
		vis[u]=1;
		if(u<=k+1||u==n)	tmp++;
		if(tmp==k+2)	break;
		for(i=head[u];i;i=next[i])	if(dis[to[i]]>dis[u]+val[i])
			dis[to[i]]=dis[u]+val[i],q.push(mp(-dis[to[i]],to[i]));
	}
	for(i=1;i<=k;i++)	v[S-1][i]=dis[i+1];
	if(!r[S-1])	f[1<<S-2][S-1]=dis[1];
	g[S-1]=dis[n];
}
int main()
{
	n=rd(),m=rd(),k=rd();
	int i,j,h,a,b,c;
	for(i=1;i<=m;i++)	a=rd(),b=rd(),c=rd(),add(a,b,c),add(b,a,c);
	if(!k)
	{
		dijkstra(1);
		printf("%d\n",dis[n]);
		return 0;
	}
	kk=rd();
	for(i=1;i<=kk;i++)	a=rd()-1,b=rd()-1,r[b]|=(1<<a-1);
	memset(f,0x3f,sizeof(f));
	for(i=2;i<=k+1;i++)	dijkstra(i);
	for(i=1;i<(1<<k);i++)
	{
		for(j=1;j<=k;j++)	if(i&(1<<j-1))	for(h=1;h<=k;h++)	if(!(i&(1<<h-1))&&(i&r[h])==r[h])
			f[i|(1<<h-1)][h]=min(f[i|(1<<h-1)][h],f[i][j]+v[j][h]);
	}
	for(ans=1<<30,i=1;i<=k;i++)	ans=min(ans,f[(1<<k)-1][i]+g[i]);
	printf("%d",ans);
	return 0;
}
时间: 2024-11-03 22:19:08

【BZOJ1097】[POI2007]旅游景点atr 最短路+状压DP的相关文章

BZOJ 1097: [POI2007]旅游景点atr( 最短路 + 状压dp )

先最短路预处理, 然后状压就行了 -------------------------------------------------------------------------- #include<cstdio> #include<cstring> #include<algorithm> #include<cctype> #include<queue> using namespace std; #define b(i) (1 <<

BZOJ1097 旅游景点atr (最短路+状压DP)

链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1097分析:见注释. 1 #include <cstdio> 2 #include <vector> 3 #include <algorithm> 4 #include <cstring> 5 #include <functional> 6 #include <queue> 7 using namespace std; 8 9 #d

BZOJ1097 [POI2007]旅游景点atr

首先预处理出来前K个点互相之间的最短路,直接Dijkstra就好了 然后就变成了状压DP...随便写一下好了 1 /************************************************************** 2 Problem: 1097 3 User: rausen 4 Language: C++ 5 Result: Accepted 6 Time:18456 ms 7 Memory:98872 kb 8 ***************************

【BZOJ-1097】旅游景点atr SPFA + 状压DP

1097: [POI2007]旅游景点atr Time Limit: 30 Sec  Memory Limit: 357 MBSubmit: 1531  Solved: 352[Submit][Status][Discuss] Description FGD想从成都去上海旅游.在旅途中他希望经过一些城市并在那里欣赏风景,品尝风味小吃或者做其他的有趣的事情.经过这些城市的顺序不是完全随意的,比如说FGD不希望在刚吃过一顿大餐之后立刻去下一个城市登山,而是希望去另外什么地方喝下午茶.幸运的是,FGD

HDU 4856 Tunnels (最短路+状压DP)

题意:给你N*N的网格,'.'表示可以走,'#'表示不能走,m条管道,每条管道有起点和终点坐标, Bob每次可以走到相邻的网格花费1s,问Bob走完m条管道要花多少时间:Bob在管道内不计算时间 即计算Bob从管道 i 的出口走到管道 j 的入口的时间Dis(e[i],s[j])的最小和,起点可以任意: 思路:看了题解说是状态压缩DP然后深入理解了下. 首先算出d[e[i]][s[j]]的最短距离,不能到达为-1: dp[i][j] : 表示以 j 为起点状态为 i 的最小值.其中 i 是用十进

【bzoj1097】[POI2007]旅游景点atr 状压dp+堆优化Dijkstra

题目描述 FGD想从成都去上海旅游.在旅途中他希望经过一些城市并在那里欣赏风景,品尝风味小吃或者做其他的有趣的事情.经过这些城市的顺序不是完全随意的,比如说FGD不希望在刚吃过一顿大餐之后立刻去下一个城市登山,而是希望去另外什么地方喝下午茶.幸运的是,FGD的旅程不是既定的,他可以在某些旅行方案之间进行选择.由于FGD非常讨厌乘车的颠簸,他希望在满足他的要求的情况下,旅行的距离尽量短,这样他就有足够的精力来欣赏风景或者是泡MM了^_^.整个城市交通网络包含N个城市以及城市与城市之间的双向道路M条

BZOJ 1097 POI2007 旅游景点atr SPFA+状压DP

题目大意:给定一张图,要求从第一个点出发,按照某个拓扑序遍历2~k+1的所有节点,然后到达n,求最短路径 首先将所有关键点之间的最短路用SPFA求出来 然后状压DP 令f[state][p]表示已经走过的点集为state,将要走到p点的最短路 记忆化搜索就行了- - 标准卡时过- - #include <cstdio> #include <cstring> #include <iostream> #include <algorithm> #define M

[POI2007]旅游景点atr

Description FGD想从成都去上海旅游.在旅途中他希望经过一些城市并在那里欣赏风景,品尝风味小吃或者做其他的有趣的事情.经过这些城市的顺序不是完全随意的,比如说FGD不希望在刚吃过一顿大餐之后立刻去下一个城市登山,而是希望去另外什么地方喝下午茶.幸运的是,FGD的旅程不是既定的,他可以在某些旅行方案之间进行选择.由于FGD非常讨厌乘车的颠簸,他希望在满足他的要求的情况下,旅行的距离尽量短,这样他就有足够的精力来欣赏风景或者是泡MM了^_^.整个城市交通网络包含N个城市以及城市与城市之间

【弱校胡策】2016.4.14 (bzoj2164)最短路+状压DP+矩阵乘法+高斯消元+树链剖分+线段树+背包DP

cyyz&qhyz&lwyz&gryz弱校胡策 命题人:cyyz ws_fqk T3暴力写挫了 50+10+0滚粗辣! 奇妙的约会(appointment.cpp/c/pas) [问题描述] DQS和sxb在网上结识后成为了非常好的朋友,并且都有着惊人 的OI水平.在NOI2333的比赛中,两人均拿到了金牌,并保送进入 HU/PKU.于是两人决定在这喜大普奔的时刻进行面基. NOI2333参赛选手众多,所以安排了n个考点,DQS在1号考点, 而sxb在n号考点.由于是举办全国性赛事