题意:

给定一个有重边有自环的无向图,n个点(2 <= n <= 100000), m条边(1 <= m <= 200000), 每条边有一个权值, 求从第一个点到n的最少步数, 如果最少步数相同有多条路径, 那么输出权值字典序最小的一条。

分析:

用BFS解决最短路问题, 可以先从终点BFS, 求出每个点到终点的最短距离。 那么最少步数就是起点的最短距离, 最短路径就是从起点每次向最短距离比自己少1的顶点移动(如果有多个则可以随便走), 这样就可以保证走的是最短路径, 如果一开始我们是从起点BFS, 那么这样则不能保证走的是通往终点的最短路径。然后我们就可以从起点出发, 循环最短距离次, 每次选择字典序最少的走,  如果有多个字典序相同则选择多个, 直到走完最短距离, 就可以得出答案。 注意两次BFS都需要添加标记, 不然重边很可能就会导致TLE。

 #include <bits/stdc++.h>
using namespace std;
const int maxm = 1e7;
const int maxn = 1e6 + ;
const int inf = 1e9;
struct Node{
int v,col,next;
Node():v(),col(),next(){}
};
struct ele{
int v;
int dist;
ele(int v, int dist):v(v),dist(dist){}
}; Node edge[maxn];
int G[maxn], d[maxn];
bool vis[maxn];
int n, m, cnt; void build(){
memset(G,-,sizeof(G));
cnt = ;
for(int i = ; i < m; i++){
int u, v, col;
scanf("%d %d %d", &u, &v, &col);
if(u == v) continue;
edge[cnt].v = v;
edge[cnt].col = col;
edge[cnt].next = G[u];
G[u] = cnt++;
edge[cnt].v = u;
edge[cnt].col = col;
edge[cnt].next = G[v];
G[v] = cnt++;
}
} void revbfs(){
fill(d,d+maxn, inf);
memset(vis,,sizeof(vis));
queue<ele> q;
q.push(ele(n,));
d[n] = ;
vis[n] = ;
while(!q.empty()){
ele u = q.front(); q.pop();
d[u.v] = u.dist;
for(int i = G[u.v]; i != -; i = edge[i].next){
int v = edge[i].v;
if(d[v] < u.dist + || vis[v]){
continue;
}
q.push(ele(v,u.dist+));
vis[v] = ;
}
}
}
void bfs(){
vector<int> path;
memset(vis,,sizeof(vis));
vis[] = ;
vector<int> next;
next.push_back();
for(int i = ; i < d[]; i++){//the essential minimum step
int min_col = inf;
for(int j = ; j < next.size(); j++){
int u = next[j];
for(int k = G[u]; k != -; k = edge[k].next){
int v = edge[k].v;
if(d[u] == d[v] + )
min_col = min(min_col,edge[k].col);
}
}
//find out the minimum color
path.push_back(min_col); vector<int> next2;
for(int j = ; j < next.size(); j++){
int u = next[j];
for(int k = G[u]; k != -; k= edge[k].next){
int v = edge[k].v;
if(d[u] == d[v] + && !vis[v] && edge[k].col == min_col){
vis[v] = ;
next2.push_back(v);
}
}
}
next = next2;
} printf("%d\n%d",(int)path.size(),path[]);
for(int i = ; i < path.size(); i++){
printf(" %d",path[i]);
}
puts("");
}
int main(){
freopen("1.txt","r",stdin);
while(~scanf("%d %d", &n, &m)){
build();
revbfs();//反向bfs求出终点到每个点的最短距离
bfs();
}
printf("%.3f",(double)clock()/CLOCKS_PER_SEC);
return ;
}

UVa 1599 理想路径(反向BFS 求最短路径 )的更多相关文章

  1. UVA 816 -- Abbott's Revenge(BFS求最短路)

     UVA 816 -- Abbott's Revenge(BFS求最短路) 有一个 9 * 9 的交叉点的迷宫. 输入起点, 离开起点时的朝向和终点, 求最短路(多解时任意一个输出即可).进入一个交叉 ...

  2. UVa 1599 Ideal Path【BFS】

    题意:给出n个点,m条边,每条边上涂有一个颜色,求从节点1到节点n的最短路径,如果最短路径有多条,要求经过的边上的颜色的字典序最小 紫书的思路:第一次从终点bfs,求出各个节点到终点的最短距离, 第二 ...

  3. UVA 1599, POJ 3092 Ideal Path 理想路径 (逆向BFS跑层次图)

    大体思路是从终点反向做一次BFS得到一个层次图,然后从起点开始依次向更小的层跑,跑的时候选则字典序最小的,由于可能有多个满足条件的点,所以要把这层满足条件的点保存起来,在跑下一层.跑完一层就会得到这层 ...

  4. Uva 1599 最佳路径

    题目链接:https://uva.onlinejudge.org/external/15/1599.pdf 题意: 保证在最短路的时候,输出字典序最小的路径. 方法: 路径上有了权值,可以利用图论的数 ...

  5. Uva 1599 Ideal Path - 双向BFS

    题目连接和描述以后再补 这题思路很简单但还真没少折腾,前后修改提交了七八次才AC...(也说明自己有多菜了).. 注意问题: 1.看清楚原题的输入输出要求,刚了书上的中文题目直接开撸,以为输入输出都是 ...

  6. 理想路径——双向BFS

    题目 给n个点m条边(2 ≤ n ≤ 100000,1 ≤ m ≤ 200000)的无向图,每条边上都涂有一种颜色.求从结点1到结点n的一条路径,使得经过的边数尽量的少,在此前提下,经过边的颜色序列的 ...

  7. BFS - 求最短路径

    Farmer John has been informed of the location of a fugitive cow and wants to catch her immediately. ...

  8. 图-用DFS求连通块- UVa 1103和用BFS求最短路-UVa816。

    这道题目甚长, 代码也是甚长, 但是思路却不是太难.然而有好多代码实现的细节, 确是十分的巧妙. 对代码阅读能力, 代码理解能力, 代码实现能力, 代码实现技巧, DFS方法都大有裨益, 敬请有兴趣者 ...

  9. UVA 1599 Ideal Path(双向bfs+字典序+非简单图的最短路+队列判重)

    https://vjudge.net/problem/UVA-1599 给一个n个点m条边(2<=n<=100000,1<=m<=200000)的无向图,每条边上都涂有一种颜色 ...

随机推荐

  1. 《Windows核心编程系列》九谈谈同步设备IO与异步设备IO之同步设备IO

    同步设备IO 所谓同步IO是指线程在发起IO请求后会被挂起,IO完成后继续执行. 异步IO是指:线程发起IO请求后并不会挂起而是继续执行.IO完毕后会得到设备的通知.而IO完成端口就是实现这种通知的很 ...

  2. 暑期训练狂刷系列——poj 3264 Balanced Lineup(线段树)

    题目连接: http://poj.org/problem?id=3264 题目大意: 有n个数从1开始编号,问在指定区间内,最大数与最小数的差值是多少? 解题思路: 在节点中存储max,min,然后查 ...

  3. 题解报告:hdu 1220 Cube

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1220 问题描述 Cowl擅长解决数学问题. 有一天,一位朋友问他这样一个问题:给你一个边长为N的立方体 ...

  4. B/S和C/S示意图

    B/S C/S

  5. Selenium2工作原理

    Selenium 经历了两个版本,Selenium 1.0 和 Selenium 2.0,本文仅介绍Selenium2的原理,在Selenium 2.0 主推的是WebDriver,Selenium2 ...

  6. SpringCloud+MyBatis+Redis整合—— 超详细实例(二)

    2.SpringCloud+MyBatis+Redis redis①是一种nosql数据库,以键值对<key,value>的形式存储数据,其速度相比于MySQL之类的数据库,相当于内存读写 ...

  7. 415 Add Strings 字符串相加

    给定两个字符串形式的非负整数 num1 和num2 ,计算它们的和.注意:    num1 和num2 的长度都小于 5100.    num1 和num2 都只包含数字 0-9.    num1 和 ...

  8. [BZOJ1878][SDOI2009]HH的项链 莫队

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1878 不带修改的莫队,用一个桶记录一下当前区间中每种颜色的数量就可以做到$O(1)$更新了 ...

  9. php(一)

    PHP (Hypertext preprocessor 超文本预处理器) 1.环境工具 Xampp等工具 2.apache配置 默认的Apache路径是  c:/xampp/apache 文件夹 可以 ...

  10. php中读取以及写入文件的方法总结

    ==>读取文件内容(方法一) $fileData = fread($fileStream,filesize($filePath)); 注意: 文本文件读取到网页上显示时,由于换行符不被解释,文本 ...