zoukankan      html  css  js  c++  java
  • SSLOJ 1297.GF打Dota

    题目

    题目描述

          众所周知,GF同学喜欢打dota,而且打得非常好。今天GF和Spartan同学进行了一场大战。现在GF拿到一张地图,地图上一共有n个地点,GF的英雄处于1号点,Spartan的基地位于n号点,GF要尽快地选择较短的路线让他的英雄去虐掉Spartan的基地。但是Spartan早就料到了这一点,他有可能会开挂(BS~)使用一种特别的魔法,一旦GF所走的路线的总长度等于最短路的总长度时,GF的英雄就要和这种魔法纠缠不休。这时GF就不得不选择非最短的路线。现在请你替GF进行规划。
    对于描述的解释与提醒:
        1.无向路径,花费时间当然为非负值。
        2.对于本题中非最短路线的定义:不管采取任何迂回、改道方式,只要GF所走的路线总长度不等于1到n最短路的总长度时,就算做一条非最短的路线。
        3.保证1~n有路可走。

    输入

    第一行为n,m(表示一共有m条路径) 
    接下来m行,每行3个整数a,b,c,表示编号为a,b的点之间连着一条花费时间为c的无向路径。 
    接下来一行有一个整数p,p=0表示Spartan没有开挂使用这种魔法,p=1则表示使用了。

    输出

    所花费的最短时间t,数据保证一定可以到达n。

    输入样例复制

    样例输入1:
    5 5
    1 2 1
    1 3 2
    3 5 2
    2 4 3
    4 5 1
    0

    样例输入2:
    5 5
    1 2 1
    1 3 2
    3 5 2
    2 4 3
    4 5 1
    1

    输出样例复制

    样例输出1:
    4
    样例输出2:
    5

    说明

    对于50%的数据,1<=n,m<=5000 
    对于70%的数据,1<=n<=10000, 1<=m<=50000,p=0, 
    对于100%的数据,1<=n<=10000, 1<=m<=50000,p=1 
    无向图,花费时间c>=0 

     

    分析

    • 很明显一个spfa就好了

    • 那我们如何求次短路呢
    • 我们可以枚举每一段路
    • 做两次spfa
    • 然后枚举每一条边(x,y),求出起点到x+终点到y+边(x,y)权值之和,若不等于1~n最短路长度则更新ans。 预期得分100分

        spfa pop之后一定要放回零

    代码

     1 #include<iostream>
     2 #include<vector>
     3 #include<queue>
     4 #include<cstring>
     5 using namespace std;
     6 int map[10010][10010];
     7 int n,m;
     8 vector <int> f[10010];
     9 int vis[10010],dis[10010],aa[50010],bb[50010],cc[50010],diss[10010];
    10 queue <int> q,qq;
    11 void spfa()
    12 {
    13     vis[1]=1; dis[1]=0; q.push(1);
    14     while (!q.empty())
    15     {
    16         int x=q.front(); q.pop(); vis[x]=0; 
    17         for (int i=0;i<f[x].size();i++)
    18         {
    19             int y=f[x][i];
    20             if (dis[x]+map[x][y]<dis[y])
    21             {
    22                 dis[y]=dis[x]+map[x][y];
    23                 if (!vis[y])
    24                 {
    25                    vis[y]=1;
    26                    q.push(y);
    27                 }
    28             }
    29         }
    30     }
    31 }
    32 void spfa1()
    33 {
    34     memset(vis,false,sizeof(vis));
    35     diss[n]=0;
    36     qq.push(n);
    37     while (!qq.empty())
    38     {
    39         int x=qq.front(); qq.pop(); vis[x]=0;
    40         for (int i=0;i<f[x].size();i++)
    41         {
    42             int y=f[x][i];
    43             if (diss[x]+map[x][y]<diss[y])
    44             {
    45                 diss[y]=diss[x]+map[x][y];
    46                 if (!vis[y])
    47                 {
    48                    vis[y]=1;
    49                    qq.push(y);
    50                 }
    51             }
    52         }
    53     }
    54 }
    55 int main ()
    56 {
    57     cin>>n>>m;
    58     for (int i=1,a,b,c;i<=m;i++)
    59     {
    60         cin>>a>>b>>c;
    61         aa[i]=a;
    62         bb[i]=b;
    63         cc[i]=c;
    64         f[a].push_back(b);
    65         f[b].push_back(a);
    66         map[a][b]=c;
    67         map[b][a]=c;
    68     }
    69     int p;
    70     cin>>p;
    71     memset(dis,0x3f,sizeof(dis));
    72     spfa();
    73     int minn=dis[n];
    74     if (p==0) {
    75         cout<<dis[n];
    76         return 0;
    77     }
    78     else
    79     {
    80         memset(diss,0x3f,sizeof(diss));
    81         spfa1();
    82         int ans=1e9;
    83         for (int i=1;i<=m;i++)
    84         {
    85             if (dis[aa[i]]+diss[bb[i]]+cc[i]!=minn)
    86               ans=min(ans,dis[aa[i]]+diss[bb[i]]+cc[i]);
    87             if (diss[aa[i]]+dis[bb[i]]+cc[i]!=minn)
    88               ans=min(ans,diss[aa[i]]+dis[bb[i]]+cc[i]);
    89         }
    90         cout<<ans;
    91     }
    92 }
    为何要逼自己长大,去闯不该闯的荒唐
  • 相关阅读:
    安装Hadoop
    爬虫综合大作业
    理解爬虫原理
    中文词频统计与词云生成
    复合数据类型,英文词频统计
    字符串操作、文件操作,英文词频统计预处理
    了解大数据的特点、来源与数据呈现方式
    结对项目-四则运算 “软件”之升级版
    分布式版本控制系统Git的安装与使用
    Maven 依赖的scope标签用法
  • 原文地址:https://www.cnblogs.com/zjzjzj/p/10461387.html
Copyright © 2011-2022 走看看