天梯赛 大区赛 L3-014.周游世界 (Dijkstra)

L3-014. 周游世界

时间限制

200 ms

内存限制

65536 kB

代码长度限制

8000 B

判题程序

Standard

作者

陈越

周游世界是件浪漫事,但规划旅行路线就不一定了…… 全世界有成千上万条航线、铁路线、大巴线,令人眼花缭乱。所以旅行社会选择部分运输公司组成联盟,每家公司提供一条线路,然后帮助客户规划由联盟内企业支持的旅行路线。本题就要求你帮旅行社实现一个自动规划路线的程序,使得对任何给定的起点和终点,可以找出最顺畅的路线。所谓“最顺畅”,首先是指中途经停站最少;如果经停站一样多,则取需要换乘线路次数最少的路线。

输入格式:

输入在第一行给出一个正整数N(<= 100),即联盟公司的数量。接下来有N行,第i行(i=1, ..., N)描述了第i家公司所提供的线路。格式为:

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

其中M(<= 100)是经停站的数量,S[i](i=1, ..., M)是经停站的编号(由4位0-9的数字组成)。这里假设每条线路都是简单的一条可以双向运行的链路,并且输入保证是按照正确的经停顺序给出的 —— 也就是说,任意一对相邻的S[i]和S[i+1](i=1, ..., M-1)之间都不存在其他经停站点。我们称相邻站点之间的线路为一个运营区间,每个运营区间只承包给一家公司。环线是有可能存在的,但不会不经停任何中间站点就从出发地回到出发地。当然,不同公司的线路是可能在某些站点有交叉的,这些站点就是客户的换乘点,我们假设任意换乘点涉及的不同公司的线路都不超过5条。

在描述了联盟线路之后,题目将给出一个正整数K(<= 10),随后K行,每行给出一位客户的需求,即始发地的编号和目的地的编号,中间以一空格分隔。

输出格式:

处理每一位客户的需求。如果没有现成的线路可以使其到达目的地,就在一行中输出“Sorry, no line is available.”;如果目的地可达,则首先在一行中输出最顺畅路线的经停站数量(始发地和目的地不包括在内),然后按下列格式给出旅行路线:

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

其中Xi是线路承包公司的编号,Si是经停站的编号。但必须只输出始发地、换乘点和目的地,不能输出中间的经停站。题目保证满足要求的路线是唯一的。

输入样例:

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 head[maxn+5];
int tot;
void add(int x,int y,int z)
{
    edge[tot].value=y;
    edge[tot].value2=x;
    edge[tot].next=head[x];
    edge[tot].id=z;
    head[x]=tot++;
}
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;
    for(int i=head[s];i!=-1;i=edge[i].next)
    {
        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;
                }
                
            }
        }
        for(int i=head[term.next];i!=-1;i=edge[i].next)
        {
            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;
    memset(head,-1,sizeof(head));
    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
            {
                add(x,y,i);
                add(y,x,i);
            }
            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 条评论
登录 后参与评论

相关文章

来自专栏平凡文摘

十年之后再看“面向对象”

1203
来自专栏vue学习

前言

underscore.js一直听说都是一个很经典的库,很适合新手入门,所以历经小半年断断续续的学习,总算是把它敲完了。然后又过了一段时间到了现在,回过头来,打算...

1041
来自专栏Albert陈凯

Scala一个综合的案例《learn scala in half an hour》 by jeff.kit

2011年1月23号Jeff参加了珠三角技术沙龙2011年1月广州小沙龙,并在会上给大家做了一个关于Scala的分享,形式是通过一个设计好的剧本(以沙龙聚会为背...

2665
来自专栏编程之旅

使用Block提高代码可读性

最近一直在思考并持续的扩充着自己的技术栈,比如每天坚持着学习前端知识,并且时常想着在移动端这条路上,自己的技术盲区。诚然,想要在一个领域达到一定的技术高度是挺困...

1093
来自专栏大数据风控

R中如何利用余弦算法实现相似文章的推荐

推荐(Recommended) 介绍好的人或事物,希望被任用或接受。在目前的数据挖掘领域, 推荐包括相似推荐以及协同过滤推荐。 相似推荐(Simil...

3245
来自专栏精讲JAVA

十年之后再看“面向对象”

一起帮里有人问“面向对象”的问题。但我创建“一起帮”的目的是帮人解决“具体的”“实务性的”问题,“面向对象”太过于抽象,所以没批准发布。后来在QQ群里讨论,看他...

1966
来自专栏程序人生

打造属于你自己的乐高积木

夫鹄不日浴而白,乌不日黔而黑 -- 庄周 上面的这句话某种程度来说是不妥的,人性(这也是全体生物进化出的本能)趋利避害,如果不施加外力,很容易走向消极的一面。...

4258
来自专栏程序人生 阅读快乐

Java编程思想-第4版

本书赢得了全球程序员的广泛赞誉,即使是最晦涩的概念,在Bruce Eckel的文字亲和力和小而直接的编程示例面前也会化解于无形。从Java的基础语法到最高级特性...

872
来自专栏企鹅号快讯

两种编程高手

第一种工程师 给一段复杂的程序,比如有7个局部变量,5层循环和if嵌套,他能赤手空拳上阵,迅速领会程序意图、找到bug,不用借助任何工具甚至纸笔。 给一个复杂的...

2055
来自专栏编舟记

架构整洁之道导读(一)

我是《架构整洁之道》(Clean Architecture) 中文版的技术审校者,在审校的过程当中略有感悟,所以希望通过撰写导读的方式分享给大家。

4338

扫码关注云+社区

领取腾讯云代金券