题目描述:

给定一张图,问从1和n相遇的最短时间。

这道题的输入比较特殊,不能直接存,所以怎么输入,怎么存取,只要可以访问到一个节点的相邻节点就可以,由于spfa算法的时间复杂度为m*n,而Dijkstra算法的时间复杂度为m*log(n),

其实Dijkstra就是用优先队列替换掉了spfa中的普通队列,每次都是用权值最小的点去更新别人,可以这么理解,用用权值最小的点a去更新其他节点,在更新节点中必然会有一个节点已经是

最短路,假设是b,因为源点到其他节点的距离+其他节点到b节点的距离 >  dist[a] + edge[a][b], 因为 源点到其他节点的距离 > dist[a] + edge[a][b];最多有10^6个点,那么可能就会有 10^12条边 ,时间复杂度为 10^12 * log(10^5) = 15*10^12;所以主要取决于有多少条边,分析一下如果一个节点已经更新了它所在集合中的其他点,那么这个集合中所有点之间的边就不会在更新了,假设a,b,c,d,e在一个集合里,集合之间的路径为edge,假设a是更新点,说明b,c,d,e的最短路大于a的最短路, 说明b,c,d的最短路一定是由a得到,而不是 b,c,d,Dis[a]+edge < Dist[k] +edge,k属于b,c,d。所以集合只会被访问一次。

//有可能一个人在家等他
#include <iostream>
#include <cstdio>
#include <cstring>
#include <vector>
#include <queue>
#include <algorithm>
#include <math.h>
#define maxn 1100000
#define LL long long
using namespace std;
int t;
int n,m;
vector < int > G[maxn];
int tt[maxn],ss[maxn];
vector < int > Set[maxn];
LL Dist[][maxn];
int vis[maxn];
int VISIT[maxn];
LL answer[maxn];
LL ans[maxn];
struct node
{
int u,dist;
friend bool operator < (node a,node b)
{
return a.dist>b.dist;
}
};
void init()
{
memset(vis,,sizeof(vis));
memset(VISIT,,sizeof(VISIT));
} void Dijkstra(int start)
{
int kind;
if(start==)
kind=;
else
kind=;
priority_queue < node > que;
node s;
s.u=start;
s.dist=;
Dist[kind][start]=;
que.push(s);
while(!que.empty())
{
s=que.top();
que.pop();
if(vis[s.u]==)
continue;
vis[s.u]=;
int from=s.u;
for(int i=;i<Set[from].size();i++) //找到所在集合
{
int j=Set[from][i]; //j代表哪个集合
if(VISIT[j]==)
continue;
VISIT[j]=;
int edge= tt[j]; //edge代表长度
for(int k=;k<G[j].size();k++) //遍历相邻节点
{
node Next;
int to=G[j][k];
if(Dist[kind][to]>Dist[kind][from]+edge || Dist[kind][to]==- )
{
Dist[kind][to]=Dist[kind][from] + edge;
Next.dist=Dist[kind][to];
Next.u=to;
que.push(Next);
}
}
}
}
} void solve()
{
init();
Dijkstra();
init();
Dijkstra(n);
for(int i=;i<=n;i++)
{
// cout<<Dist[1][i]<<" "<<Dist[2][i]<<endl;
if(Dist[][i]==- || Dist[][i]==-)
answer[i]=-;
else
answer[i]=max(Dist[][i],Dist[][i]);
}
LL MIN=-;
for(int i=;i<=n;i++)
{
if(MIN > answer[i] || (MIN==-) )
MIN=answer[i];
}
if(MIN==-)
printf("Evil John\n");
else
{
printf("%I64d\n",MIN);
int flag=;
for(int i=;i<=n;i++)
{
if(MIN==answer[i])
{
if(flag==)
printf(" ");
printf("%d",i);
flag=;
}
} printf("\n");
}
}
void input()
{
scanf("%d%d",&n,&m);
for(int i=;i<=m;i++)
{
scanf("%d%d",&tt[i],&ss[i]);
for(int j=;j<=ss[i];j++)
{
int input;
scanf("%d",&input);
G[i].push_back(input);
Set[input].push_back(i);
}
}
}
void Init()
{
for(int i=;i<maxn;i++)
{
G[i].clear(); Set[i].clear();
}
memset(Dist,-,sizeof(Dist));
}
int main()
{
//freopen("test.txt","r",stdin);
scanf("%d",&t);
int Case=;
// printf("%d\n",inf);
// cout<<inf<<endl;
while(t--)
{
Init();
input();
printf("Case #%d: ",++Case);
solve();
}
return ;
}

hdu5521 ( Dijkstra )的更多相关文章

  1. HDU5521 Meeting(dijkstra+巧妙建图)

    HDU5521 Meeting 题意: 给你n个点,它们组成了m个团,第i个团内有si个点,且每个团内的点互相之间距离为ti,问如果同时从点1和点n出发,最短耗时多少相遇 很明显题目给出的是个无负环的 ...

  2. Dijkstra 单源最短路径算法

    Dijkstra 算法是一种用于计算带权有向图中单源最短路径(SSSP:Single-Source Shortest Path)的算法,由计算机科学家 Edsger Dijkstra 于 1956 年 ...

  3. 最短路径算法-Dijkstra

    Dijkstra是解决单源最短路径的一般方法,属于一种贪婪算法. 所谓单源最短路径是指在一个赋权有向图中,从某一点出发,到另一点的最短路径. 以python代码为例,实现Dijkstra算法 1.数据 ...

  4. [板子]最小费用最大流(Dijkstra增广)

    最小费用最大流板子,没有压行.利用重标号让边权非负,用Dijkstra进行增广,在理论和实际上都比SPFA增广快得多.教程略去.转载请随意. #include <cstdio> #incl ...

  5. POJ 2253 Frogger(Dijkstra)

    传送门 Frogger Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 39453   Accepted: 12691 Des ...

  6. POJ 2387 Til the Cows Come Home(最短路 Dijkstra/spfa)

    传送门 Til the Cows Come Home Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 46727   Acce ...

  7. Dijkstra 算法

    all the nodes should be carectorized into three groups: (visited, front, unknown) we should pay spec ...

  8. 51nod 1459 迷宫游戏 (最短路径—Dijkstra算法)

    题目链接 中文题,迪杰斯特拉最短路径算法模板题. #include<stdio.h> #include<string.h> #define INF 0x3f3f3f3f ],v ...

  9. 51nod1459(带权值的dijkstra)

    题目链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1459 题意:中文题诶- 思路:带权值的最短路,这道题数据也没 ...

随机推荐

  1. HDU 1426 dancing links解决数独问题

    题目大意: 这是一个最简单的数独填充题目,题目保证只能产生一种数独,所以这里的初始9宫格较为稠密,可以直接dfs也没有问题 但最近练习dancing links,这类数据结构解决数独无疑效率会高很多 ...

  2. mysql免安装版配置使用

    mysql免安装版配置使用 1.下载解压 2.配置环境变量 变量MYSQL_HOME  = 解压目录 配置变量path 编辑,在后面加上  ;%MYSQL_HOME%\bin 3.修改配置文件 增加或 ...

  3. Codevs 2666 2666 Accept Ratio

    时间限制: 1 s  空间限制: 32000 KB   题目等级 : 钻石 Diamond 题目描述 Description 某陈痴迷于pku的ACM题库,常常彻夜奋斗刷题.他最近的目标是在NOIP0 ...

  4. URAL 1614. National Project “Trams” [ 构造 欧拉回路 ]

    传送门 1614. National Project “Trams” Time limit: 0.5 secondMemory limit: 64 MB President has declared ...

  5. linux shell if语句使用方法 [转载]

    最精简的 if 命令的语法是: if TEST-COMMANDS; then CONSEQUENT-COMMANDS; fi TEST-COMMAND 执行后且它的返回状态是0,那么 CONSEQUE ...

  6. HDU 5883 欧拉路径异或值最大 水题

    The Best Path Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others)Tot ...

  7. 最长上升子序列(LIS)长度的O(nlogn)算法

    最长上升子序列(LIS)的典型变形,熟悉的n^2的动归会超时.LIS问题可以优化为nlogn的算法.定义d[k]:长度为k的上升子序列的最末元素,若有多个长度为k的上升子序列,则记录最小的那个最末元素 ...

  8. php monolog 的写日志到unix domain socket 测试终于成功

    在另外一个客户端执行 php s.php后, 通过nc -lU /tmp/tg.sck 建立的unix domain socket 有接收到消息. <?php require 'vendor/a ...

  9. HDU 1201 18岁生日 【日期】

    18岁生日 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Subm ...

  10. SQL 主机

    SQL 主机 SQL 主机 如果您想要您的网站存储数据在数据库并从数据库显示数据,您的 Web 服务器必须能使用 SQL 语言访问数据库系统. 如果您的 Web 服务器托管在互联网服务提供商(ISP, ...