# L3-014. 周游世界

200 ms

65536 kB

8000 B

Standard

M S[1] S[2] ... S[M]

Go by the line of company #X1 from S1 to S2.
Go by the line of company #X2 from S2 to S3.
......

4
7 1001 3212 1003 1204 1005 1306 7797
9 9988 2333 1204 2006 2005 2004 2003 2302 2001
13 3011 3812 3013 3001 1306 3003 2333 3066 3212 3008 2302 3010 3011
4 6666 8432 4011 1306
4
3011 3013
6666 2001
2004 3001
2222 6666

2
Go by the line of company #3 from 3011 to 3013.
10
Go by the line of company #4 from 6666 to 1306.
Go by the line of company #3 from 1306 to 2302.
Go by the line of company #2 from 2302 to 2001.
6
Go by the line of company #2 from 2004 to 1204.
Go by the line of company #1 from 1204 to 1306.
Go by the line of company #3 from 1306 to 3001.
Sorry, no line is available.
pat里的最短路，一般都是条件很多的，又是路线，又是换乘的。这道题目用Dijkstra的时候应该注意，如果一般的Dijkstra以点为节点插入优先队列
那么会有这样的情况：到达同一个点的最优线路有两条，那么你就要再开一个数组记录并列的最优线路有多少条。如果以边为节点插入优先队列，那么就不存在
多条最优路线的情况，
#include <iostream>
#include <string.h>
#include <stdlib.h>
#include <algorithm>
#include <stdio.h>
#include <math.h>
#include <string>
#include <queue>

using namespace std;
const int maxn=1e5;
typedef long long int LL;
const int INF=1e9;
struct Node
{
int value;
int next;
int value2;
int id;
}edge[maxn*4+5],a[maxn+5];
int tot;
{
edge[tot].value=y;
edge[tot].value2=x;
edge[tot].id=z;
}
struct node
{
int pre;
int next;
int id;
int num;
int num2;
int pos;
node(){};
node(int pre,int next,int id,int num,int num2,int pos)
{
this->pre=pre;
this->next=next;
this->id=id;
this->num=num;
this->num2=num2;
this->pos=pos;
}
friend bool operator <(node a,node b)
{
return a.num>b.num;
}
};
int d[maxn+5];//站点数19958
int v[maxn+5];//换乘次数
int n,m;
int route[maxn+5];
int vis[maxn+5];
int Dijkstra(int s,int e)
{
priority_queue<node> q;
int res1=INF;
int res2=INF;
int ans=-1;
memset(vis,0,sizeof(vis));
for(int i=0;i<=tot;i++)
d[i]=v[i]=INF;
{
d[i]=1;v[i]=0;
route[i]=-1;
vis[i]=1;
q.push(node(s,edge[i].value,edge[i].id,d[i],v[i],i));
}
while(!q.empty())
{
node term=q.top();
q.pop();
vis[term.pos]=1;
if(term.next==e)
{
if(res1>term.num)
{
ans=term.pos;
res1=term.num;
res2=term.num2;
}
else if(res1==term.num)
{
if(res2>term.num2)
{
res2=term.num2;
ans=term.pos;
}

}
}
{
int x=edge[i].value;
if(vis[i]) continue;
if(d[i]>term.num+1)
{
d[i]=term.num+1;
if(term.id!=edge[i].id)
v[i]=term.num2+1;
else
v[i]=term.num2;
route[i]=term.pos;
q.push(node(term.next,edge[i].value,edge[i].id,d[i],v[i],i));
}
else if(d[i]==term.num+1)
{
int xx;
if(term.id!=edge[i].id)
xx=term.num2+1;
else
xx=term.num2;
if(v[i]>xx)
{
v[i]=xx;
route[i]=term.pos;
q.push(node(term.next,edge[i].value,edge[i].id,d[i],v[i],i));

}
}
}
}
return ans;
}
int main()
{
scanf("%d",&n);
int k;
tot=0;
for(int i=1;i<=n;i++)
{
scanf("%d",&k);
int x,y=-1;
for(int j=1;j<=k;j++)
{
scanf("%d",&x);
if(y==-1)
{
y=x;
continue;
}
else
{
}
y=x;
}
}
scanf("%d",&m);
int x,y;
for(int i=1;i<=m;i++)
{
scanf("%d%d",&x,&y);
//if(x==y)
//{printf("0\n");continue;}
int res=Dijkstra(x, y);
if(res==-1)
{printf("Sorry, no line is available.\n");continue;}

printf("%d\n",d[res]);
int cot=0;
while(res!=-1)
{
a[++cot]=edge[res];
res=route[res];
}
int p=-1;
int tag=0;
while(cot>=2)
{
if(a[cot].id!=p)
{
if(p!=-1)
printf(" to %04d.\n",a[cot].value2);
printf("Go by the line of company #%d from %04d",a[cot].id,a[cot].value2);
p=a[cot].id;
}
cot--;
tag=1;
}
if(a[cot].id!=p)
{
if(tag)
printf(" to %04d.\n",a[cot].value2);
printf("Go by the line of company #%d from %04d to %04d.\n",a[cot].id,a[cot].value2,a[cot].value);

}
else
printf(" to %04d.\n",a[cot].value);

}
return 0;
}

0 条评论

• ### LeetCode Contest 166

但是在用c++的快排的时候，被坑了，我一直的习惯是写自定义比较函数，没有写运算符重载，不知道为什么一直RE，浪费了挺多时间

• ### LeetCode 164. Maximum Gap (排序)

题解：首先，当然我们可以用快排，排完序之后，遍历一遍数组，就能得到答案了。但是快速排序的效率是O(n* logn)，不是题目要求的线性效率，也就是O(n)的效率...

• ### 【PAT乙级】数组元素循环右移问题

版权声明：本文为博主原创文章，遵循 CC 4.0 BY-SA 版权协议，转载请附上原文出处链接和本声明。 ...

• ### 一遍记住Java常用的八种排序算法与代码实现

(如果每次比较都交换，那么就是交换排序；如果每次比较完一个循环再交换，就是简单选择排序。)

• ### HDU 1573 X问题

Problem Description 求在小于等于N的正整数中有多少个X满足： Input 输入数据的第一行为一个正整数T，表示有T组测试数据。每组测...

• ### Day2上午解题报告

预计分数：100+0+60=160 实际分数：100+0+60=160 mmpT1数据错了。。。 T1遭遇 题目描述 你是能看到第一题的 friends呢。 —...

• ### LeetCode 第 210 场周赛 解题报告

那么在遍历过程中，栈中元素数量的最大值即为答案。栈中的(可以理解为还没遍历到匹配的)，即那些嵌套的(。

• ### 洛谷P4180 [Beijing2010组队]次小生成树Tree

题目描述 小C最近学了很多最小生成树的算法，Prim算法、Kurskal算法、消圈算法等等。正当小C洋洋得意之时，小P又来泼小C冷水了。小P说，让小C求出一个无...

• ### 图论--拓扑排序--判断一个图能否被拓扑排序

拓扑排序的实现条件，以及结合应用场景，我们都能得到拓扑排序适用于DAG图（Directed Acyclic Graph简称DAG）有向无环图， 根据关系我们能得...