Gym - 100570B :ShortestPath Query(SPFA及其优化)
题意:给定N点M边的有向图,每条边有距离和颜色,一条有效路径上不能有相邻的边颜色相同。现在给定起点S,多次讯问S到点X的最短有效距离。
TLE思路:用二维状态dis(u,c)表示起点到u,最后一条边的颜色是c的最短距离,用map解决了二维空间不足的问题。但是T第151个点。所以需要优化,标解的优化是把dis(u,c)改为dis(u,0)或者dis(u,1)分别表示颜色同和不同的最短距离。
其他思路:用边表示通过那条边到点的最短距离。由于边自带颜色属性,所以就不需要其他状态了。这个方法比较直观,可以直接看代码。
TLE代码:(加了优化也不行。
#include<bits/stdc++.h>
#define rep(i,a,b) for(int i=a;i<=b;i++)
#define ll long long
#define pii pair<int,int>
#define mp make_pair
#define Fi first
#define Se second
using namespace std;
const int maxn=;
const ll inf=1LL<<;
int Laxt[maxn],Next[maxn],To[maxn],W[maxn],C[maxn];
int S,N,cnt; ll ans[maxn];
deque<pii>q;
map<int,ll>dis[maxn];
map<int,int>vis[maxn];
void add(int u,int v,int w,int c){
Next[++cnt]=Laxt[u];Laxt[u]=cnt;To[cnt]=v; W[cnt]=w; C[cnt]=c;
}
void spfa()
{
rep(i,,N) ans[i]=inf; ans[S]=;
for(int i=Laxt[S];i;i=Next[i]){
dis[To[i]][C[i]]=W[i]; ans[To[i]]=W[i];
vis[To[i]][C[i]]=;
if(!q.empty()&&dis[To[i]][C[i]]<dis[q.front().Fi][q.front().Se])
q.push_front(mp(To[i],C[i]));
else q.push_back(mp(To[i],C[i]));
}
while(!q.empty()){
pii H=q.front(); q.pop_front();
int u=H.Fi,c=H.Se; ll tdis=dis[u][c]; vis[u][c]=;
for(int i=Laxt[u];i;i=Next[i]){
int v=To[i];
if(c==C[i]) continue;
if(dis[v].find(C[i])==dis[v].end()){
dis[v][C[i]]=tdis+W[i];
ans[v]=min(ans[v],dis[v][C[i]]);
if(!vis[v][C[i]]){
vis[v][C[i]]=;
if(!q.empty()&&dis[v][C[i]]<dis[q.front().Fi][q.front().Se]) q.push_front(mp(v,C[i]));
else q.push_back(mp(v,C[i]));
}
}
else if(tdis+W[i]<dis[v][C[i]]){
dis[v][C[i]]=tdis+W[i];
ans[v]=min(ans[v],dis[v][C[i]]);
if(!vis[v][C[i]]){
vis[v][C[i]]=;
if(!q.empty()&&dis[v][C[i]]<dis[q.front().Fi][q.front().Se]) q.push_front(mp(v,C[i]));
else q.push_back(mp(v,C[i]));
}
}
}
}
rep(i,,N) if(ans[i]==inf) ans[i]=-;
}
int main()
{
int M,CC,Q,u,v,w,c;
scanf("%d%d%d",&N,&M,&CC);
rep(i,,M){
scanf("%d%d%d%d",&u,&v,&w,&c);
add(u,v,w,c);
}
scanf("%d%d",&S,&Q);
spfa();
while(Q--){
scanf("%d",&u);
printf("%I64d\n",ans[u]);
}
return ;
}
AC代码:
#include<bits/stdc++.h>
#define rep(i,a,b) for(int i=a;i<=b;i++)
#define ll long long
using namespace std;
const int maxn=;
const ll inf=1LL<<;
int Laxt[maxn],Next[maxn],To[maxn],W[maxn],C[maxn];
int S,N,cnt,vis[maxn]; ll ans[maxn],dis[maxn];
void add(int u,int v,int w,int c){
Next[++cnt]=Laxt[u];Laxt[u]=cnt;To[cnt]=v; W[cnt]=w; C[cnt]=c;
}
void spfa()
{
deque<int>q;
rep(i,,N) ans[i]=inf; ans[S]=;
rep(i,,cnt) dis[i]=inf;
for(int i=Laxt[S];i;i=Next[i]){
dis[i]=W[i]; q.push_back(i); vis[i]=;
}
while(!q.empty()){
int u=q.front(); q.pop_front(); vis[u]=;
for(int i=Laxt[To[u]];i;i=Next[i]){
if(C[u]!=C[i]){
if(dis[i]>dis[u]+W[i]){
dis[i]=dis[u]+W[i];
if(!vis[i]) {
vis[i]=;
if(!q.empty()&&dis[i]<dis[q.front()]) q.push_front(i); //注意不要为空
else q.push_back(i);
}
}
}
}
}
rep(i,,cnt) ans[To[i]]=min(ans[To[i]],dis[i]);
rep(i,,N) if(ans[i]==inf) ans[i]=-;
}
int main()
{
int M,CC,Q,u,v,w,c;
scanf("%d%d%d",&N,&M,&CC);
rep(i,,M){
scanf("%d%d%d%d",&u,&v,&w,&c);
add(u,v,w,c);
}
scanf("%d%d",&S,&Q);
spfa();
while(Q--){
scanf("%d",&u);
printf("%I64d\n",ans[u]);
}
return ;
}
Gym - 100570B :ShortestPath Query(SPFA及其优化)的更多相关文章
- codeforce Gym 100570B 	ShortestPath Query (最短路SPFA)
		题意:询问单源最短路径,每条边有一个颜色,要求路径上相邻边的颜色不能相同,无重边且边权为正. 题解:因为路径的合法性和边的颜色有关, 所以在做spfa的时候,把边丢到队列中去,松弛的时候注意判断一下颜 ... 
- SPFA 小优化*2
		/* bzoj 2763 SPFA小优化 循环队列+SLF 顺面改掉自己之前手打qeueu的坏毛病*/ #include<iostream> #include<cstring> ... 
- HDU 1535 Invitation Cards(SPFA,及其优化)
		题意: 有编号1-P的站点, 有Q条公交车路线,公交车路线只从一个起点站直接到达终点站,是单向的,每条路线有它自己的车费. 有P个人早上从1出发,他们要到达每一个公交站点, 然后到了晚上再返回点1. ... 
- [BZOJ 2200][Usaco2011 Jan]道路和航线 spfa+SLF优化
		Description Farmer John正在一个新的销售区域对他的牛奶销售方案进行调查.他想把牛奶送到T个城镇 (1 <= T <= 25,000),编号为1T.这些城镇之间通过R条 ... 
- 【最短路径】 SPFA算法优化
		首先先明确一个问题,SPFA是什么?(不会看什么看,一边学去,传送门),SPFA是bellman-ford的队列优化版本,只有在国内才流行SPFA这个名字,大多数人就只知道SPFA就是一个顶尖的高效算 ... 
- SPFA队列优化
		spfa队列优化(用来求最短路) 实现方法: 1.存入图.可以使用链式前向星或者vocter. 2.开一个队列,先将开始的节点放入. 3.每次从队列中取出一个节点X,遍历与X相通的Y节点,查询比对 ... 
- spfa + slf优化
		最近在练习费用流 , 不是要用spfa吗 ,我们教练说:ns学生写朴素的spfa说出去都让人笑 . QwQ,所以就去学了一下优化 . slf优化就是双向队列优化一下,本来想用lll优化,可是优化后我t ... 
- 最短路--spfa+队列优化模板
		spfa普通版就不写了,优化还是要的昂,spfa是可以判负环,接受负权边和重边的,判断负环只需要另开一个数组记录每个结点的入队次数,当有任意一个结点入队大于点数就表明有负环存在 #include< ... 
- CDOJ 1287 MC挖矿世界(Spfa+set优化)
		题目大意:原题链接 解题思路:此题要求多点最短距离,但是直接套用floyd会超时. 然后我们想直接从每一个点开始bfs就好了,但是还是会TLE,为什么呢? 因为你访问了很多次没有意义的地方,因为有些点 ... 
随机推荐
- ios - 使用@try、catch捕获异常:
			@try { // 可能会出现崩溃的代码 } @catch (NSException *exception) { // 捕获到的异常exception } @finally { // 结果处理 } 
- Spring 和 filter
			标题是 spring和filter,但是这里却是说的spring MVC 项目中需要用到filter,filter中需要用到spring实例化的bean,于是为了简化就形成spring和filter了 ... 
- CentOS7.1安装 Vsftpd FTP 服务器
			# yum install vsftpd 安装 Vsftpd FTP 编辑配置文件 ‘/etc/vsftpd/vsftpd.conf’ 用于保护 vsftpd. # vi /etc/vsftpd/vs ... 
- Django 之restframework1
			Restframework 这里先简单的介绍一下restful协议 ----一切皆是资源,操作只是请求方式 基于restful协议的框架有很多Django下的restframework只是其中的一种 ... 
- 队列(Queue)
			队列(Queue) Queue: 先入先出(FIFO)的数据结构. offer,add区别: 一些队列有大小限制,因此如果想在一个满的队列中加入一个新项,多出的项就会被拒绝. 这时新的 offer 方 ... 
- Linux中cp命令不提示直接覆盖的方法
			新做了服务器,cp覆盖时,无论加什么参数-f之类的还是提示是否覆盖,这在大量cp覆盖操作的时候是不能忍受的. 把a目录下的文件复制到b目录 cp –r a/* b 执行上面的命令时,b存在的每个文件都 ... 
- QCon2016 上海会议汇总(1) - 前端技术实践
			日程传送门:http://2016.qconshanghai.com/schedule 我这里重点总结下前端.移动端.团队管理.研发支撑相关的议题,谈谈我的感受. <Vue 2.0: 渐进式前端 ... 
- iOS  发大招  otherButtonTitles:(nullable NSString *)otherButtonTitles, ...  写法   && 编写通用类的时候关于可变参数的处理
			开始 我 以为 这个 alertView 里面 ...的写法 应该 是一个 普通的数组 然 并没有 分享一篇好文 http://www.tekuba.net/program/290/ IOS实现 ... 
- 【LeetCode】数组排列问题(permutations)(附加next_permutation解析)
			描述 Given a collection of distinct integers, return all possible permutations. Example: Input: [1,2,3 ... 
- ubuntu下android studio生成的unaligned apk的zipalign处理
			在ubuntu系统中使用android studio生成的apk文件始终都是unaligned apk, 在bulid.gradle中设置如下设置后,还是同样生成的是unaligned apk. mi ... 
