HUST 1408
来源:互联网 发布:压缩软件 mac 编辑:程序博客网 时间:2024/06/05 06:30
http://acm.hust.edu.cn/problem/show/1408
[转] Floyd 算法原理
floyd算法是一个经典的动态规划算法。用通俗的语言来描述的话,首先我们的目标是寻找从点i到点j的最短路径。从动态规划的角度看问题,我们需要为这个目标重新做一个诠释(这个诠释正是动态规划最富创造力的精华所在),floyd算法加入了这个概念
Ak(i,j):表示从i到j中途不经过索引比k大的点的最短路径。
这个限制的重要之处在于,它将最短路径的概念做了限制,使得该限制有机会满足迭代关系,这个迭代关系就在于研究:假设Ak(i,j)已知,是否可以借此推导出Ak-1(i,j)。
假设我现在要得到Ak(i,j),而此时Ak(i,j)已知,那么我可以分两种情况来看待问题:1. Ak(i,j)沿途经过点k;2. Ak(i,j)不经过点k。如果经过点k,那么很显然,Ak(i,j) = Ak-1(i,k) + Ak-1(k,j),为什么是Ak-1呢?因为对(i,k)和(k,j),由于k本身就是源点(或者说终点),加上我们求的是Ak(i,j),所以满足不经过比k大的点的条件限制,且已经不会经过点k,故得出了Ak-1这个值。那么遇到第二种情况,Ak(i,j)不经过点k时,由于没有经过点k,所以根据概念,可以得出Ak(i,j)=Ak-1(i,j)。现在,我们确信有且只有这两种情况---不是经过点k,就是不经过点k,没有第三种情况了,条件很完整,那么是选择哪一个呢?很简单,求的是最短路径,当然是哪个最短,求取哪个,故得出式子:
Ak(i,j) = min( Ak-1(i,j), Ak-1(i,k) + Ak-1(k,j) )
因此floyd的最外层循环:
for (k = 0; k < n; k++) ...
就是分别求出 A0(i,j), A1(i,j), ..., An(i,j)
我屡次写错floyd的程序,今天又写错一次。。尽管它很短,但原理真的很牛比。
只要知道了原理,就不会再写错了!
#include<cstdio>#include<string>#include<cstring>#include<iostream>#include<cmath>#include<algorithm>#include<climits>#include<queue>#include<vector>#include<map>#include<sstream>#include<set>#include<stack>using namespace std;typedef long long ll;//ifstream fin("input.txt");//ofstream fout("output.txt");//fin.close();//fout.close();ll dp[110][110];ll L1, L2, L3, L4, C1, C2, C3, C4;int n,m;ll x[110];const ll INF=999999999999;void init(){int i,j;for(i=1;i<=n;i++){for(j=1;j<=n;j++){if(i==j) dp[i][j]=0;else if( 0<abs( x[j]-x[i])&&abs( x[j]-x[i])<=L1 ) dp[i][j]=C1;else if( L1<abs(x[j]-x[i])&&abs(x[j]-x[i])<=L2 ) dp[i][j]=C2;else if( L2<abs(x[j]-x[i])&&abs(x[j]-x[i])<=L3 ) dp[i][j]=C3;else if(L3<abs(x[j]-x[i])&&abs(x[j]-x[i])<=L4 ) dp[i][j]=C4;else dp[i][j]=INF;}}}void floyd(){int i,j,k;for(k=1;k<=n;k++){for(i=1;i<=n;i++){for(j=1;j<=n;j++){dp[i][j]=min(dp[i][j],dp[i][k]+dp[k][j]);}}} }int main(){int i,j,T,u,v,kase=0;scanf("%d",&T);while(T--){printf("Case %d:\n",++kase);scanf("%lld%lld%lld%lld%lld%lld%lld%lld",&L1, &L2, &L3, &L4, &C1,& C2, &C3, &C4);// cout<<L1<<L2<<L3<<L4<<C1<<C2<<C3<<C4<<endl;scanf("%d%d",&n,&m);for(i=1;i<=n;i++){scanf("%lld",&x[i]);}init();floyd();// for(i=1;i<=n;i++)// for(j=1;j<=n;j++)// cout<<dp[i][j]<<endl;while(m--){scanf("%d%d",&u,&v);if(dp[u][v]>=INF)printf("Station %d and station %d are not attainable.\n",u,v);elseprintf("The minimum cost between station %d and station %d is %lld.\n",u,v,dp[u][v]);}} return 0;}
- HUST 1408
- HUST 1408 公交系统
- HUST 1408 公交系统
- HUST
- HUST
- HUST
- HUST
- HUST 1328
- HUST-1010
- hust 1602
- HUST 1525
- hust 1010
- HUST 1422
- HUST 1563
- hust 1592
- HUST 1017
- HUST 1017
- hust 1007 平衡树
- bootstrap3 input 输入框图标背景
- jquery全选,jquery全不选,jquery反选
- 高精度乘法
- FZU1404 Andy的作业
- POJ 3349
- HUST 1408
- IOS支付宝集成一系列问题
- chmod命令详解
- 求二叉树的深度和宽度平衡最大距离
- PAT-PAT (Advanced Level) Practise 1001. A+B Format (20) 【二星级】
- scala学习第四章 映射和元组
- 15.8.19笔记
- Oracle ORA-01940 无法删除当前已连接用户
- TCP/IP协议三次握手与四次握手流程解析