• PTA天梯地图


    本题要求你实现一个天梯赛专属在线地图,队员输入自己学校所在地和赛场地点后,该地图应该推荐两条路线:一条是最快到达路线;一条是最短距离的路线。题目保证对任意的查询请求,地图上都至少存在一条可达路线。

    输入格式:

    输入在第一行给出两个正整数N(2 N ≤ 500)和M,分别为地图中所有标记地点的个数和连接地点的道路条数。随后M行,每行按如下格式给出一条道路的信息:

    V1 V2 one-way length time
    

    其中V1V2是道路的两个端点的编号(从0到N-1);如果该道路是从V1V2的单行线,则one-way为1,否则为0;length是道路的长度;time是通过该路所需要的时间。最后给出一对起点和终点的编号。

    输出格式:

    首先按下列格式输出最快到达的时间T和用节点编号表示的路线:

    Time = T: 起点 => 节点1 => ... => 终点
    

    然后在下一行按下列格式输出最短距离D和用节点编号表示的路线:

    Distance = D: 起点 => 节点1 => ... => 终点
    

    如果最快到达路线不唯一,则输出几条最快路线中最短的那条,题目保证这条路线是唯一的。而如果最短距离的路线不唯一,则输出途径节点数最少的那条,题目保证这条路线是唯一的。

    如果这两条路线是完全一样的,则按下列格式输出:

    Time = T; Distance = D: 起点 => 节点1 => ... => 终点
    

    输入样例1:

    10 15
    0 1 0 1 1
    8 0 0 1 1
    4 8 1 1 1
    5 4 0 2 3
    5 9 1 1 4
    0 6 0 1 1
    7 3 1 1 2
    8 3 1 1 2
    2 5 0 2 2
    2 1 1 1 1
    1 5 0 1 3
    1 4 0 1 1
    9 7 1 1 3
    3 1 0 2 5
    6 3 1 2 1
    5 3
    

    输出样例1:

    Time = 6: 5 => 4 => 8 => 3
    Distance = 3: 5 => 1 => 3
    

    输入样例2:

    7 9
    0 4 1 1 1
    1 6 1 3 1
    2 6 1 1 1
    2 5 1 2 2
    3 0 0 1 1
    3 1 1 3 1
    3 2 1 2 1
    4 5 0 2 2
    6 5 1 2 1
    3 5
    

    输出样例2:

    Time = 3; Distance = 4: 3 => 2 => 5

    解题思路:当时建图的时候把距离搞成了点的距离,和城市间紧急救援搞混了,看了一晚上自己的代码都没看出来,最后请教了Bob才知道,大佬的博客https://www.cnblogs.com/BobHuang/
    菜鸡的成长史 ^_^
    代码有点乱等以后来改进
      1 #include <bits/stdc++.h>
      2 using namespace std;
      3 const int N=505,INF=0x3f3f3f3f;
      4 int n,m;
      5 struct edg
      6 {
      7     int B,J,S;
      8 }E1,E2;    //存边的信息,因为这个一条边有3个信息
      9 vector<edg>G[N],T[N];
     10 int dis[N],pre[N],vis[N],in[N],num[N],out[N];
     11 int diss[N],sum[N];
     12 struct Node
     13 {
     14     int B,J;
     15     bool operator <(const Node&x)const{
     16         return x.J<J;
     17     }
     18 }p,q;
     19 void printff(vector<int> vec)
     20 {
     21     int flag=0;
     22     for(int i=vec.size()-1;i>=0;i--)
     23     {
     24         if(flag) cout << " => ";
     25         cout << vec[i],flag=1;
     26     }
     27     cout << endl;
     28 }
     29 int judge(vector<int>vec,vector<int>vec1)
     30 {
     31     for(int i=0;i<vec.size();i++)
     32         if(vec[i]!=vec1[i]) return 0;    //不相等就退出
     33     return 1;
     34 }
     35 void dij1(int start)
     36 {
     37     priority_queue<Node> que;
     38     for(int i=0;i<=N-1;i++) dis[i]=INF,pre[i]=-1,vis[i]=0;
     39     que.push({start,0}),dis[start]=0,num[start]=1;
     40     while(!que.empty())
     41     {
     42         p=que.top(),que.pop();
     43         int u=p.B,v,w,z;
     44         if(vis[u]) continue;
     45         vis[u]=1;
     46         for(auto X:G[u])
     47         {
     48             v=X.B,w=X.J,z=1;
     49             if(dis[v]>w+p.J)
     50                 dis[v]=w+p.J,pre[v]=u,num[v]=num[u]+1,q.B=v,q.J=dis[v],que.push(q);
     51             else if(dis[v]==w+p.J)
     52             {
     53                 if(num[v]>num[u]+1)  //节点数取小的
     54                     num[v]=num[u]+1,pre[v]=u;
     55             }
     56         }
     57     }
     58 }
     59 void dij2(int start)    //时间加距离
     60 {
     61     priority_queue<Node> que;
     62     for(int i=0;i<=N-1;i++) pre[i]=-1,vis[i]=0,sum[i]=INF,diss[i]=0;
     63     que.push({start,0}),diss[start]=0,sum[start]=0; //自己到自己的距离和时间都为0
     64     while(!que.empty())
     65     {
     66         p=que.top(),que.pop();
     67         int u=p.B,v,w,z;
     68         if(vis[u]) continue;
     69         vis[u]=1;
     70         for(auto X:G[u])
     71         {
     72             v=X.B,w=X.J,z=X.S;  //压入的是时间
     73             if(sum[v]>z+p.J)
     74                 sum[v]=z+p.J,pre[v]=u,diss[v]=diss[u]+w,q.B=v,q.J=sum[v],que.push(q);
     75             else if(sum[v]==z+p.J)
     76             {
     77                 if(diss[v]>diss[u]+w)  //节点数取小的
     78                     diss[v]=diss[u]+w,pre[v]=u;
     79             }
     80         }
     81     }
     82 }
     83 
     84 int main()
     85 {
     86     ios::sync_with_stdio(false);
     87     cin>>n>>m;
     88     for(int i=0,d1,d2,d3,d4,d5;i<m;i++)
     89     {
     90         cin>>d1>>d2>>d3>>d4>>d5;
     91         G[d1].push_back({d2,d4,d5}); //G 最短中最快
     92         if(!d3)         //双行线
     93         {
     94             G[d2].push_back({d1,d4,d5});
     95         }
     96     }
     97     vector<int> vec,vec1;
     98     int start,ending;
     99     cin>>start>>ending;
    100     dij2(start);     //时间+距离
    101     int pos=ending;
    102     while(pos!=-1) vec.push_back(pos),pos=pre[pos];  //vec为时间的
    103     dij1(start);    //距离+节点
    104     pos=ending;
    105     while(pos!=-1) vec1.push_back(pos),pos=pre[pos];
    106     if(vec.size()!=vec1.size())
    107     {
    108         cout << "Time = " << sum[ending] <<": ";
    109         printff(vec);
    110         cout << "Distance = " << dis[ending] <<": ";
    111         printff(vec1);
    112     }
    113     else
    114     {
    115         int zhi=judge(vec,vec1);   //0不相等 1相等
    116         if(zhi==1)
    117             cout<<"Time = "<<sum[ending]<<"; "<<"Distance = "<<dis[ending]<<": ",printff(vec);
    118         else
    119         {
    120             cout << "Time = " << sum[ending] <<": ";
    121             printff(vec);
    122             cout << "Distance = " << dis[ending] <<": ";
    123             printff(vec1);
    124         }
    125     }
    126     return 0;
    127 }
    View Code

  • 相关阅读:
    C# 微信品牌会员卡开发(微信会员卡2.0)
    管理者问卷调查
    二:elementui源码解析之改造demoblock可以直接在卡片里编辑修改代码并生效渲染到界面上
    MySql 的@符号定义一个变量在sql里的占位符作用
    swift 代码段的重构
    k8skubeadm高可用安装部署
    LeetCode> 71. 简化路径
    Linux进程管理
    Linux中断下半部及推后执行的工作
    Linux进程调度
  • 原文地址:https://www.cnblogs.com/qq-1585047819/p/10606750.html
Copyright © 2020-2023  润新知