CSU 1808 - 地铁 - [最短路变形]
题目链接:http://acm.csu.edu.cn/csuoj/problemset/problem?pid=1808
Time limit: 5000 ms Memory limit: 131072 kB
Bobo 居住在大城市 ICPCCamp。
Input
Output
对于每组数据,输出一个整数表示要求的值。
Sample Input
3 3
1 2 1 1
2 3 2 1
1 3 1 1
3 3
1 2 1 1
2 3 2 1
1 3 1 10
3 2
1 2 1 1
2 3 1 1
Sample Output
1
3
2
题解:
如果只记录到某个节点x的最短路长度d[x],并且记录对应于d[x],是坐哪号线来到节点x的,这样显然是错误的。
原因比如这样的样例:
3 3
1 2 1 2
1 2 3 3
2 3 3 5
可以看出,d[x]要扩展到d[x][c],即这题的状态有两个量决定:到了节点x,最后乘坐的是c号线;
那么,如果我们把节点x用若干条边Edge(u1→x)…Edge(uk→x)来代替,那么我们就相当于把d[x]要扩展到d[x][c]了;
所以我们可以直接把边当成点,对边做最短路。
(这题对边做最短路,如果用spfa的话会TLE,要使用堆优化dijkstra)
AC代码:
#include<cstdio>
#include<cmath>
#include<cstring>
#include<iostream>
#include<vector>
#include<algorithm>
#include<queue>
using namespace std;
typedef long long LL;
typedef pair<LL,int> Pair; const LL INF=1e18;
const int maxn=1e5+;
const int maxm=2e5+; //无向边拆成两条有向边 int n,m; struct Edge{
int u,v,c;
int next;
LL t;
};
Edge E[maxm];
int head[maxn],ne;
void init()
{
ne=;
memset(head,-,sizeof(head));
}
void addedge(int u,int v,int c,LL t)
{
E[ne].u=u, E[ne].v=v, E[ne].c=c, E[ne].t=t;
E[ne].next=head[u];
head[u]=ne++;
} LL ans;
LL d[maxm];
bool vis[maxm];
void dijkstra(int st)
{
priority_queue< Pair, vector<Pair>, greater<Pair> > Q; memset(vis,,sizeof(vis));
for(int i=;i<ne;i++) d[i]=INF;
ans=INF; for(int i=head[st];i!=-;i=E[i].next)
{
d[i]=E[i].t;
Q.push(Pair(d[i],i));
}
while(!Q.empty())
{
int x=Q.top().second; Q.pop(); if(vis[x]) continue;
vis[x]=;
if(E[x].v==n) ans=min(ans,d[x]); for(int y=head[E[x].v];y!=-;y=E[y].next)
{
if(vis[y]) continue;
if(d[y]>d[x]+E[y].t+abs(E[y].c-E[x].c))
{
d[y]=d[x]+E[y].t+abs(E[y].c-E[x].c);
Q.push(Pair(d[y],y));
}
}
}
}
int main()
{
while(scanf("%d%d",&n,&m)!=EOF)
{
init();
for(int i=;i<=m;i++)
{
int u,v,c; LL t;
scanf("%d%d%d%lld",&u,&v,&c,&t);
addedge(u,v,c,t);
addedge(v,u,c,t);
} dijkstra();
printf("%lld\n",ans);
}
}
#include<cstdio>
#include<cmath>
#include<cstring>
#include<iostream>
#include<vector>
#include<algorithm>
#include<queue>
using namespace std;
typedef long long LL;
typedef pair<LL,int> Pair; const LL INF=1e18;
const int maxn=1e5+;
const int maxm=2e5+; //无向边拆成两条有向边 int n,m; struct Edge{
int u,v,c;
LL t;
};
vector<Edge> E;
vector<int> G[maxn];
void init(int l,int r)
{
E.clear();
for(int i=l;i<=r;i++) G[i].clear();
}
void addedge(int u,int v,int c,LL t)
{
E.push_back((Edge){u,v,c,t});
G[u].push_back(E.size()-);
} LL ans;
LL d[maxm];
bool vis[maxm];
void dijkstra(int st)
{
priority_queue< Pair, vector<Pair>, greater<Pair> > Q; memset(vis,,sizeof(vis));
for(int i=;i<E.size();i++) d[i]=INF;
ans=INF; for(int i=;i<G[st].size();i++)
{
int x=G[st][i];
d[x]=E[x].t;
Q.push(Pair(d[x],x));
}
while(!Q.empty())
{
int x=Q.top().second; Q.pop(); if(vis[x]) continue;
vis[x]=;
if(E[x].v==n) ans=min(ans,d[x]); for(int i=;i<G[E[x].v].size();i++)
{
int y=G[E[x].v][i];
if(vis[y]) continue;
if(d[y]>d[x]+E[y].t+abs(E[y].c-E[x].c))
{
d[y]=d[x]+E[y].t+abs(E[y].c-E[x].c);
Q.push(Pair(d[y],y));
}
}
}
}
int main()
{
while(scanf("%d%d",&n,&m)!=EOF)
{
init(,n);
for(int i=;i<=m;i++)
{
int u,v,c; LL t;
scanf("%d%d%d%lld",&u,&v,&c,&t);
addedge(u,v,c,t);
addedge(v,u,c,t);
} dijkstra();
printf("%lld\n",ans);
}
}
注:两份代码的区别是分别用链式前向星和vector邻接表存图。
CSU 1808 - 地铁 - [最短路变形]的更多相关文章
- CSU 1808: 地铁 最短路
题目链接: http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1808 1808: 地铁 Time Limit: 5 SecMemory Limit: ...
- CSU 1808 地铁(最短路变形)
http://acm.csu.edu.cn/csuoj/problemset/problem?pid=1808 题意: Bobo 居住在大城市 ICPCCamp. ICPCCamp 有 n 个地铁站, ...
- 【最短路】【STL】CSU 1808 地铁 (2016湖南省第十二届大学生计算机程序设计竞赛)
题目链接: http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1808 题目大意: N个点M条无向边(N,M<=105),每条边属于某一条地铁Ci ...
- CSU 1808 地铁
题意: ICPCCamp 有 n 个地铁站,用 1,2,-,n 编号. m 段双向的地铁线路连接 n 个地铁站,其中第 i 段地铁属于 ci 号线,位于站 ai,bi 之间,往返均需要花费 ti 分钟 ...
- CSU 1808 地铁 (Dijkstra)
Description Bobo 居住在大城市 ICPCCamp. ICPCCamp 有 n 个地铁站,用 1,2,-,n 编号. m 段双向的地铁线路连接 n 个地铁站,其中第 i 段地铁属于 ci ...
- POJ-2253.Frogger.(求每条路径中最大值的最小值,最短路变形)
做到了这个题,感觉网上的博客是真的水,只有kuangbin大神一句话就点醒了我,所以我写这篇博客是为了让最短路的入门者尽快脱坑...... 本题思路:本题是最短路的变形,要求出最短路中的最大跳跃距离, ...
- POJ 3635 - Full Tank? - [最短路变形][手写二叉堆优化Dijkstra][配对堆优化Dijkstra]
题目链接:http://poj.org/problem?id=3635 题意题解等均参考:POJ 3635 - Full Tank? - [最短路变形][优先队列优化Dijkstra]. 一些口胡: ...
- POJ 3635 - Full Tank? - [最短路变形][优先队列优化Dijkstra]
题目链接:http://poj.org/problem?id=3635 Description After going through the receipts from your car trip ...
- POJ-1797Heavy Transportation,最短路变形,用dijkstra稍加修改就可以了;
Heavy Transportation Time Limit: 3000MS Memory Limit: 30000K Description Background Hugo ...
随机推荐
- %s %d %f 等等是什么意思
这个是C语言的格式化输出:%s是字符串.%d是整数.%f代表浮点数. 这些是格式声明,格式声明由“%”和格式字符组成.常用的格式字符有:1)d格式符,用来输出一个有符号的十进制整数:2)c格式符,用来 ...
- 个人Android开发习惯[转载]
以下内容转载自牛人博客,很多牛叉内容,请点击:http://www.stay4it.com/?p=281 Android三年,现在写代码,真心没什么很大的兴趣,每天都是优化代码,优化框架,技术支持.非 ...
- Hibernate的Configuration和SessionFactiory
Configuration: Configuration是hibernate的入口,负责管理Hibernate的配置信息,这些配置信息都是从配置文件hibernate.cfg.xml或者Hiberna ...
- backbone学习笔记:模型(Model)(1)基础知识
backbone为复杂Javascript应用程序提供MVC(Model View Controller)框架,框架里最基本的是Model(模型),它用来处理数据,对数据进行验证,完成后台数据与前台数 ...
- Spring getBean 首字母大小写问题
如果类第一个字母大写第二个小写,那么首字母小写获取bean 如果第一个和第二个字母都是大写的,那个获取bean首字母要大写
- OSG添加回调更新
class CB : public osg::NodeCallback { virtual void operator()(osg::Node* node, osg::NodeVisitor* nv) ...
- Flask中向前端传递或者接收Json文件的方法
1. 利用flask的request.form.get()方法 这一中方法主要利用flask的request.form.get方法,获得前端发送给后台的json文件 Python 端代码: @app. ...
- MQTT_DEMO
1 /* 2 Copyright (c) 2009-2012 Roger Light <roger@atchoo.org> 3 All rights reserved. 4 5 Redis ...
- linux C 调用shell程序执行
#include<stdio.h> #include <unistd.h> #include <sys/types.h> #include <stdlib.h ...
- 《C++ Primer Plus》16.2 智能指针模板类
智能指针是行为类似于指针的类对象,单这种对象还有其他功能.本节介绍三个可帮助管理动态内存分配的智能指针类.先来看看需要哪些功能以及这些功能是如何实现的.请看下面的函数:void remodel(std ...