首页 > 代码库 > BZOJ2324: [ZJOI2011]营救皮卡丘

BZOJ2324: [ZJOI2011]营救皮卡丘

2324: [ZJOI2011]营救皮卡丘

Time Limit: 10 Sec  Memory Limit: 256 MB
Submit: 1359  Solved: 522
[Submit][Status]

Description

皮卡丘被火箭队用邪恶的计谋抢走了!这三个坏家伙还给小智留下了赤果果的挑衅!为了皮卡丘,也为了正义,小智和他的朋友们义不容辞的踏上了营救皮卡丘的道路。

火箭队一共有N个据点,据点之间存在M条双向道路。据点分别从1N标号。小智一行K人从真新镇出发,营救被困在N号据点的皮卡丘。为了方便起见,我们将真新镇视为0号据点,一开始K个人都在0号点。

由于火箭队的重重布防,要想摧毁K号据点,必须按照顺序先摧毁1K-1号据点,并且,如果K-1号据点没有被摧毁,由于防御的连锁性,小智一行任何一个人进入据点K,都会被发现,并产生严重后果。因此,在K-1号据点被摧毁之前,任何人是不能够经过K号据点的。

为了简化问题,我们忽略战斗环节,小智一行任何一个人经过K号据点即认为K号据点被摧毁。被摧毁的据点依然是可以被经过的。

K个人是可以分头行动的,只要有任何一个人在K-1号据点被摧毁之后,经过K号据点,K号据点就被摧毁了。显然的,只要N号据点被摧毁,皮卡丘就得救了。

野外的道路是不安全的,因此小智一行希望在摧毁N号据点救出皮卡丘的同时,使得K个人所经过的道路的长度总和最少。

请你帮助小智设计一个最佳的营救方案吧!

 

Input

第一行包含三个正整数N,M,K。表示一共有N+1个据点,分别从0N编号,以及M条无向边。一开始小智一行共K个人均位于0号点。 

接下来M行,每行三个非负整数,第i行的整数为Ai,Bi,Li。表示存在一条从Ai号据点到Bi号据点的长度为Li的道路。

Output

仅包含一个整数S,为营救皮卡丘所需要经过的最小的道路总和。

Sample Input


3 4 2

0 1 1

1 2 1

2 3 100

0 3 1


Sample Output

3

【样例说明】

小智和小霞一起前去营救皮卡丘。在最优方案中,小智先从真新镇前往1号点,接着前往2号据点。当小智成功摧毁2号据点之后,小霞从真新镇出发直接前往3号据点,救出皮卡丘。

HINT

 



对于10%的数据满足 K = 1,且N = 3,小智将独自前去营救皮卡丘;


对于20%的数据满足 K ≤ 3,且N ≤ 20,被小智单挑剿灭的火箭队加强了防御,增加了据点数;


对于40%的数据满足 K ≤ 3,且N ≤ 100,面对加强的防御,小智拉来了好朋友小霞和小刚,一同前去营救;


对于另外20%的数据满足任意一对据点之间均存在道路,并且对任意的0 ≤ X,Y,Z ≤ N,有不等式L(X,Z) ≤ L(X,Y) + L(Y,Z)成立;


对于100%的数据满足N ≤ 150, M ≤ 20 000, 1 ≤ K ≤ 10, Li ≤ 10 000, 保证小智一行一定能够救出皮卡丘。


至于为什么K ≤ 10,你可以认为最终在小智的号召下,小智,小霞,小刚,小建,小遥,小胜,小光,艾莉丝,天桐,还有去日本旅游的黑猫警长,一同前去大战火箭队。

 

Source

Day2

题解:

这题和星际竞速还有打印机两题的主体思路都是一样的

每个点一定要经过,并且要经过这个点,必须经过比这个点小的所有点。而且还存在一个附加源,但源到附加源有一定的容量限制(星际没有。。。)

这题我们采用如下方式建图:

1.把每个点拆成 i 和 i+n 两个点,分别表示从这个点出发和进入这个点

2.由s向所有i 连容量为1,费用为0的边

2.由所有i+n到t连容量为1,费用为0的边

3.由 i 向所有 j+n(j>n)连容量为1,费用为从 i 到 j,不经过比j标号大的中间节点的最短路 的边 (否则这条道路将不合法)

正确性可以从i+n 入流的来源来考虑,每一种流法都代表着一种实实在在的、合法的方案,cost就是花费时间,我们要时间最短,自然要最小费用最大流了

还有一个问题就是   

费用为从 i 到 j,不经过比j标号大的中间节点的最短路  怎么求?

我自己yy了一种想法,如下:

    for(int k=0;k<=n;k++)     for(int i=0;i<=n;i++)      for(int j=k;j<=n;j++)       f[i][j]=min(f[i][j],f[i][k]+f[k][j]);

意思就是不使用比 j 大的节点作为中间节点来更新 i(1..n)到j的最短路

所以这个过程结束后,f[i][j]就代表着从 i 到 j不经过比 j大的节点的最短路

代码:

 1 #include<cstdio> 2 #include<cstdlib> 3 #include<cmath> 4 #include<cstring> 5 #include<algorithm> 6 #include<iostream> 7 #include<vector> 8 #include<map> 9 #include<set>10 #include<queue>11 #define inf 100000000012 #define maxn 50013 #define maxm 10000014 #define eps 1e-1015 #define ll long long16 #define pa pair<int,int>17 using namespace std;18 inline int read()19 {20     int x=0,f=1;char ch=getchar();21     while(ch<0||ch>9){if(ch==-)f=-1;ch=getchar();}22     while(ch>=0&&ch<=9){x=10*x+ch-0;ch=getchar();}23     return x*f;24 }25 int n,m,k,tot=1,s,t,ss,head[maxn],q[maxn],from[maxn],f[maxn][maxn];26 bool v[maxn];27 ll d[maxn],mincost;28 struct edge{int from,next,go,v,c;}e[2*maxm];29 void ins(int x,int y,int z,int w)30 {31     e[++tot].go=y;e[tot].from=x;e[tot].v=z;e[tot].c=w;e[tot].next=head[x];head[x]=tot;32 }33 void insert(int x,int y,int z,int w)34 {35     ins(x,y,z,w);ins(y,x,0,-w);36 }37 bool spfa()38 {39     for (int i=s;i<=ss;i++)d[i]=inf;40     memset(v,0,sizeof(v));41     int l=0,r=1,y;q[1]=s;d[s]=0;v[0]=1;42     while(l!=r)43     {44         int x=q[++l];if(l==maxn)l=0;v[x]=0;45         for (int i=head[x];i;i=e[i].next)46          if(e[i].v&&d[x]+e[i].c<d[y=e[i].go])47          {48             d[y]=d[x]+e[i].c;from[y]=i;49             if(!v[y]){v[y]=1;q[++r]=y;if(r==maxn)r=0;}50          }51     }52     return d[t]!=inf;53 }54 void mcf()55 {56     while(spfa())57     {58         int tmp=inf;59         for(int i=from[t];i;i=from[e[i].from]) tmp=min(tmp,e[i].v);60         mincost+=d[t]*tmp;61         for(int i=from[t];i;i=from[e[i].from]){e[i].v-=tmp;e[i^1].v+=tmp;}62     }63 }64 int main()65 {66     freopen("input.txt","r",stdin);67     freopen("output.txt","w",stdout);68     n=read();m=read();k=read();69     memset(f,60,sizeof(f));70     for(int i=1;i<=m;i++)71     {72         int x=read(),y=read(),z=read();73         f[x][y]=min(f[x][y],z);f[y][x]=min(f[y][x],z);74     }75     s=0;t=2*n+1;ss=2*n+2;insert(s,ss,k,0);76     for(int k=0;k<=n;k++)77      for(int i=0;i<=n;i++)78       for(int j=k;j<=n;j++)79        f[i][j]=min(f[i][j],f[i][k]+f[k][j]);80     for(int i=1;i<=n;i++)insert(s,i,1,0);81     for(int i=1;i<=n;i++)insert(ss,i+n,1,f[0][i]);82     for(int i=1;i<=n;i++)insert(i+n,t,1,0);83     for(int i=1;i<=n;i++)84      for(int j=i+1;j<=n;j++)85       insert(i,j+n,1,f[i][j]);86     mcf();87     printf("%lld\n",mincost);     88     return 0;89 }
View Code

 

 

BZOJ2324: [ZJOI2011]营救皮卡丘