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 }

 

转载于:https://www.cnblogs.com/zjzjzj/p/10461387.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值