题目链接:http://poj.org/problem?id=2449

Time Limit: 4000MS Memory Limit: 65536K

Description

"Good man never makes girls wait or breaks an appointment!" said the mandarin duck father. Softly touching his little ducks' head, he told them a story.

"Prince Remmarguts lives in his kingdom UDF – United Delta of Freedom. One day their neighboring country sent them Princess Uyuw on a diplomatic mission."

"Erenow, the princess sent Remmarguts a letter, informing him that she would come to the hall and hold commercial talks with UDF if and only if the prince go and meet her via the K-th shortest path. (in fact, Uyuw does not want to come at all)"

Being interested in the trade development and such a lovely girl, Prince Remmarguts really became enamored. He needs you - the prime minister's help!

DETAILS: UDF's capital consists of N stations. The hall is numbered S, while the station numbered T denotes prince' current place. M muddy directed sideways connect some of the stations. Remmarguts' path to welcome the princess might include the same station twice or more than twice, even it is the station with number S or T. Different paths with same length will be considered disparate.

Input

The first line contains two integer numbers N and M (1 <= N <= 1000, 0 <= M <= 100000). Stations are numbered from 1 to N. Each of the following M lines contains three integer numbers A, B and T (1 <= A, B <= N, 1 <= T <= 100). It shows that there is a directed sideway from A-th station to B-th station with time T.

The last line consists of three integer numbers S, T and K (1 <= S, T <= N, 1 <= K <= 1000).

Output

A single line consisting of a single integer number: the length (time required) to welcome Princess Uyuw using the K-th shortest path. If K-th shortest path does not exist, you should output "-1" (without quotes) instead.

Sample Input

2 2
1 2 5
2 1 4
1 2 2

Sample Output

14

题意:

给出 $N$ 个顶点,$M$ 条有向边,求从起点 $S$ 到终点 $T$ 的第 $K$ 短路的长度,路径允许重复经过点和边。

(如果不存在第 $k$ 短路则输出 $-1$。其实不难知道如果从 $S$ 到 $T$ 的某条路径上存在一个环,则必然存在第 $k$ 短路,否则就有可能没有第 $k$ 短路)

(有一个需要注意的点是,此处当 $S=T$ 时,认为最短路径不是零)

题解:

根据优先队列优化Dijkstra算法,我们知道,优先队列里面的可能存在多个元素 $(x,d_1[x]),(x,d_2[x]), \cdots$,同时代表着某个节点 $x$ 以及起点到 $x$ 的某条路径的长度。

当从优先队列中第一次取出 $x$ 时,起点到 $x$ 的最短路长度已经得到,那么后续再取出节点 $x$ 显然就是一些次短路径。

由此我们可以想到使用优先队列BFS

起初优先队列中只存在元素 $(S,0)$,不断从优先队列中取出 $d[x]$ 最小的 $(x,d[x])$,然后沿着从 $x$ 出发的每条有向边 $edge(x,y)$ 都进行扩展,将新的元素 $(y,d[y]=d[x]+w(x,y))$ 入队(本题允许重复经过点和边)。

不难证明,某个状态 $(x,d_1[x])$ 第一次从优先队列中被取出时,就得到了从初态到它的最小代价 $d[x]$。因为此时队列中所有其他状态的代价都大于等于 $d[x]$,是不可能再返回来更新 $d[x]$ 的。

然后,如果按照普通的优先队列BFS做法(即优先队列优化Dijkstra算法),往后若再从队列内取出另一个元素 $(x,d_2[x])$,会将其直接忽略,因为必然不是最优解。

那么,不难想到在本题的优先队列BFS做法中,第 $i$ 次从优先队列中取出的元素 $(x,d_i[x])$,其 $d_i[x]$ 便是从起点到 $x$ 的第 $i$ 短路长度。

所以,最终当终点 $T$ 从队列中第 $K$ 次出队时,便求得了答案。

这样算法最坏的时间复杂度为 $O(K(N+M)log(N+M))$。

MLE代码:

#include<queue>
#include<cstdio>
#include<vector>
#include<cstring>
#include<iostream>
using namespace std;
typedef pair<int,int> pii;
const int maxn=;
int n,m,S,T,K; struct Edge{
int v,w;
Edge(){}
Edge(int _v,int _w) {
v=_v, w=_w;
}
};
vector<Edge> E[maxn]; priority_queue< pii, vector<pii>, greater<pii> > Q;
int cnt[maxn];
int bfs()
{
memset(cnt,,sizeof(cnt));
Q.push(make_pair(,S));
while(!Q.empty())
{
int x=Q.top().second, d=Q.top().first; Q.pop(); cnt[x]++;
if(cnt[T]==K) return d;
for(int k=;k<E[x].size();k++)
{
int y=E[x][k].v, w=E[x][k].w;
if(cnt[y]<K) Q.push(make_pair(d+w,y));
}
}
return -;
} int main()
{
cin>>n>>m;
for(int i=,u,v,w;i<=m;i++)
{
scanf("%d%d%d",&u,&v,&w);
E[u].push_back(Edge(v,w));
}
cin>>S>>T>>K;
if(S==T) K++;
cout<<bfs()<<endl;
}

交了一发,得到MLE $\times 1$,应该是优先队列内存的元素太多了……

A*优化:

我们知道,A*用一个估价函数 $f(x)$ 来估计当前状态到目标状态所需代价。

在本题中,实际代价 $g(x)$ 即代表:第 $k$ 短路中从 $x$ 到 $T$ 的实际距离。

显然,相应的我们可以设 $f(x)$ 代表:最短路中从 $x$ 到 $T$ 的实际距离。

这样一来,必然满足条件 $f(x) \le g(x)$,正确性得以保证。

优先队列内原本存储的是 $(x,d[x])$,现在变为 $(x,d[x]+f(x))$。

A*算法的时间复杂度上界一样是 $O(K(N+M)log(N+M))$,但由于估价函数 $f(x)$ 的作用,很多节点访问次数远小于 $k$,是有较明显的加速的(相应的优先队列内存储的元素会大幅度减少,避免MLE)。

AC代码:

#include<queue>
#include<cstdio>
#include<vector>
#include<cstring>
#include<iostream>
using namespace std;
typedef pair<int,int> pii;
const int maxn=;
const int INF=0x3f3f3f3f;
int n,m,S,T,K; struct Edge{
int v,w;
Edge(){}
Edge(int _v,int _w) {
v=_v, w=_w;
}
};
vector<Edge> E[maxn],fE[maxn];
priority_queue< pii, vector<pii>, greater<pii> > Q; int f[maxn];
bool vis[maxn];
void dijkstra()
{
memset(f,0x3f,sizeof(f));
memset(vis,,sizeof(vis));
f[T]=;
Q.push(make_pair(,T));
while(Q.size())
{
int u=Q.top().second; Q.pop();
if(vis[u]) continue;
vis[u]=;
for(int i=;i<fE[u].size();i++)
{
int v=fE[u][i].v, w=fE[u][i].w;
if(vis[v]) continue;
if(f[v]>f[u]+w)
{
f[v]=f[u]+w;
Q.push(make_pair(f[v],v));
}
}
}
} int cnt[maxn];
int bfs()
{
memset(cnt,,sizeof(cnt));
Q.push(make_pair(,S));
while(!Q.empty())
{
int x=Q.top().second, d=Q.top().first; Q.pop(); cnt[x]++;
if(cnt[T]==K) return d;
for(int k=;k<E[x].size();k++)
{
int y=E[x][k].v, w=E[x][k].w;
if(cnt[y]<K) Q.push(make_pair(d+w,y));
}
}
return -;
} int A_star()
{
memset(cnt,,sizeof(cnt));
Q.push(make_pair(+f[S],S));
while(Q.size())
{
int u=Q.top().second;
int d=Q.top().first-f[u];
Q.pop(); cnt[u]++;
if(cnt[T]==K) return d;
for(int i=,v,w;i<E[u].size();i++)
{
v=E[u][i].v, w=E[u][i].w;
if(cnt[v]<K) Q.push(make_pair(d+w+f[v],v));
}
}
return -;
} int main()
{
cin>>n>>m;
for(int i=,u,v,w;i<=m;i++)
{
scanf("%d%d%d",&u,&v,&w);
E[u].push_back(Edge(v,w));
fE[v].push_back(Edge(u,w));
}
cin>>S>>T>>K;
if(S==T) K++;
dijkstra();
cout<<A_star()<<endl;
}

POJ 2449 - Remmarguts' Date - [第k短路模板题][优先队列BFS]的更多相关文章

  1. poj 2449 Remmarguts' Date (k短路模板)

    Remmarguts' Date http://poj.org/problem?id=2449 Time Limit: 4000MS   Memory Limit: 65536K Total Subm ...

  2. poj 2449 Remmarguts' Date 第k短路 (最短路变形)

    Remmarguts' Date Time Limit: 4000MS   Memory Limit: 65536K Total Submissions: 33606   Accepted: 9116 ...

  3. poj 2449 Remmarguts' Date(K短路,A*算法)

    版权声明:本文为博主原创文章.未经博主同意不得转载. https://blog.csdn.net/u013081425/article/details/26729375 http://poj.org/ ...

  4. POJ 2449 Remmarguts' Date ( 第 k 短路 && A*算法 )

    题意 : 给出一个有向图.求起点 s 到终点 t 的第 k 短路.不存在则输出 -1 #include<stdio.h> #include<string.h> #include ...

  5. 【POJ】2449 Remmarguts' Date(k短路)

    http://poj.org/problem?id=2449 不会.. 百度学习.. 恩. k短路不难理解的. 结合了a_star的思想.每动一次进行一次估价,然后找最小的(此时的最短路)然后累计到k ...

  6. 【POJ】2449.Remmarguts' Date(K短路 n log n + k log k + m算法,非A*,论文算法)

    题解 (搬运一个原来博客的论文题) 抱着板题的心情去,结果有大坑 就是S == T的时候也一定要走,++K 我发现按照论文写得\(O(n \log n + m + k \ log k)\)算法没有玄学 ...

  7. [poj2449]Remmarguts' Date(K短路模板题,A*算法)

    解题关键:k短路模板题,A*算法解决. #include<cstdio> #include<cstring> #include<algorithm> #includ ...

  8. K短路模板POJ 2449 Remmarguts' Date

      Time Limit: 4000MS   Memory Limit: 65536K Total Submissions:32863   Accepted: 8953 Description &qu ...

  9. poj 2449 Remmarguts' Date(第K短路问题 Dijkstra+A*)

    http://poj.org/problem?id=2449 Remmarguts' Date Time Limit: 4000MS   Memory Limit: 65536K Total Subm ...

随机推荐

  1. Hadoop2.2.0分布式安装配置详解[3/3]

    测试启动 按照下面的每一步执行,执行完一定要看输出的信息,注意warn或error或fatal的情况.因为这都是可能是问题出现的地方.出现一个问题,不解决,可能就会影响接下来的测试.这才是真正的工作量 ...

  2. Eureka微服务ID

    Instance ID用于唯一标识注册到Eureka Server上的微服务实例.我们可在Eureka Server的首页直观地看到各个微服务的Instance ID.例如,图11-1中的itmuch ...

  3. [转]Linux性能分析工具汇总合集

    出于对Linux操作系统的兴趣,以及对底层知识的强烈欲望,因此整理了这篇文章.本文也可以作为检验基础知识的指标,另外文章涵盖了一个系统的方方面面.如果没有完善的计算机系统知识,网络知识和操作系统知识, ...

  4. 在 Visual Studio 生成项目时,会发现一些 dll 并没有被复制到输出目录,导致最终程序的执行错误

    发现与解决 检查了一下项目文件,发现是因为这些 dll 文件的引用其中一个叫做 嵌入互操作类型(EmbedInteropTypes)的属性被设为了 True,此时 复制本地 属性会被强制设为 Fals ...

  5. 《Essential C++》读书笔记 之 C++编程基础

    <Essential C++>读书笔记 之 C++编程基础 2014-07-03 1.1 如何撰写C++程序 头文件 命名空间 1.2 对象的定义与初始化 1.3 撰写表达式 运算符的优先 ...

  6. C#学习笔记(31)——委托窗体传值

    说明(2017-11-23 19:31:53): 1. 关于委托和窗体传值,一下午在网上查阅了大量资料,基本就是CSDN的论坛和博客园的文章,大家都在举例子,烧水.鸿门宴,看评论说还看到过沙漠足球的, ...

  7. Odoo 去掉 恼人的 "上午"和"下午"

  8. 【原】使用Json作为Python和C#混合编程时对象转换的中间文件

    一.Python中自定义类对象json字符串化的步骤[1]   1. 用 json 或者simplejson 就可以: 2.定义转换函数: 3. 定义类 4. 生成对象 5.dumps执行,引入转换函 ...

  9. visio操作

    1.上下标:选中要成为上标的文字,ctrl+shift+"=" 选中要成为下标的文字,ctrl+"="

  10. 解决java.sql.SQLException: The server time zone value 'Öйú±ê׼ʱ¼ä' is unrecognized or represents more than one time zone

    使用spring boot整合MySQL时一直报 java.sql.SQLException: The server time zone value 'Öйú±ê׼ʱ¼ä' is unrecog ...