博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
UVa1599 Ideal Path(双向bfs+字典序+非简单图的最短路+队列判重)
阅读量:6837 次
发布时间:2019-06-26

本文共 2273 字,大约阅读时间需要 7 分钟。

题目大意:

对于一个n个房间m条路径的迷宫(Labyrinth)(2<=n<=100000, 1<=m<=200000),每条路径上都涂有颜色,颜色取值范围为1<=c<=10^9。求从节点1到节点n的一条路径,使得经过的边尽量少,在这样的前提下,如果有多条路径边数均为最小,则颜色的字典序最小的路径获胜。一条路径可能连接两个相同的房间,一对房间之间可能有多条路径。输入保证可以从节点1到达节点n。

 更多细节可以参考原题:

分析:

  1. 从题目中我们可以看出,题目中的无向图是可以出现自环和重边的,自环我们可以在输入的时候检查并排除,但是重边我们需要保留,并从中选择颜色最小的边。
  2. 题目的数据量很大,不可能采用邻接矩阵存储图,因此应采用邻接表,且邻接表便于进行bfs
  3. 路径的颜色不代表路径的权重,本题中路径是无权的  

 

思路:

从终点开始倒着bfs一次,得到每个点到终点的距离,然后从起点开始,按照每次距离减1的方法寻找接下来的点的编号。按照颜色最小的走,如果有多个颜色最小,则都拉入队列中,将最小的颜色记录在res数组中。

其中,index=d[0]-d[u]就得到了当前u节点对应的距离,也就是步骤数。

 

细节:

  1. 已经进入队列的节点不能重复入队,否则复杂度太高,会tle(重复入队的复杂度至少是O(n^2),在n=100000的情况下直接tle)
  2. 第一次bfs和第二次bfs的终止时机不同,第一次找到起点就终止,第二次则是从队列中取出节点时才能终止,为的是遍历完所有导向终点且路径长度一致的边,只有这样才能结果正确
  3. d数组记录每个节点到终点n的距离,不能用0进行初始化,而终点处的初始化必须是0
  4. d数组不能不初始化,否则对于多输入题目,前面的输入可能影响后面的输出

代码实现如下:

1 #include
2 #include
3 #include
4 #include
5 using namespace std; //min()函数 6 #define max 100000 7 #define inf 0x7fffffff 8 typedef struct ver{ 9 int num, color;//边的另一端的节点编号 和 颜色 10 ver(int n,int c):num(n),color(c){}11 }Ver;12 int n,m,a,b,c;13 int d[max],res[max];//d记录每个点到终点的最短距离 res记录最短路的颜色14 bool vis[max],inqueue[max];//vis每个节点是否被访问过 inqueue标记节点是否加入了队列,防止重复加入 15 vector
edge[max];//邻接表记录图 16 void bfs(int start,int end){17 memset(inqueue,0,n);18 memset(vis,0,n);19 int u,v,c;20 queue
q;21 q.push(start);22 if(start==0){ //用于正向bfs 23 memset(res,0,sizeof(int)*n);24 while(!q.empty()){25 u=q.front();q.pop();vis[u]=1;26 if(u==n-1)return;27 int minc=inf,len=edge[u].size();28 for(int i=0;i

 

 

收获:

这是第一次学习bfs遍历复杂图,对于重边和自环的处理也终于有了一点经验,积累了自己的bfs最短路的模板

另外,UVa上的数据并不是完全可靠,对于用0初始化数组d的行为,可以用这组数据测试出wa的结果:

Input:

4 3

1 2 1

1 3 1

1 4 7

Output:

1

7

但是我实验发现,如果用0对数组d进行初始化,在UVa上仍能AC,不过我已经给UVa写信报告了这个bug,不知道他们会不会做修正。

不论如何,这道题还是收获很大滴~接下来是

 

反向bfs寻找最短路的模板

注意:d数组初始化应该用-1,并将d[n-1]=0,否则就会出现上述UVa的bug

这份代码假设在输入的时候重边已经被排除否则这份代码还需要加入u!=v的判断

代码如下:

1 void rbfs(){ 2     memset(inqueue,0,sizeof(inqueue)); 3     memset(vis,0,sizeof(vis)); 4     queue
q;q.push(n-1); 5 while(!q.empty()){ 6 u=q.front();q.pop();vis[u]=1; 7 for(int i=0,len=edge[u].size();i

 

今天就到这里啦~再见呦(●'◡'●)~~撒花撒花*★,°*:.☆\( ̄▽ ̄)/$:*.°★*

 

转载于:https://www.cnblogs.com/luruiyuan/p/5807430.html

你可能感兴趣的文章
正则表达式 ip地址
查看>>
使用ndk编译c可执行程序
查看>>
一种计算e的方法
查看>>
与Jquery Mobile的第一次亲密接触
查看>>
Windows 8实例教程系列 - 开篇
查看>>
C# 多重overide
查看>>
安装arcgis server 10.2遇到的问题总结
查看>>
查看他人数据接口的安全校验机制
查看>>
react 通过 classnames 处理 多个class 的问题
查看>>
倒计时原理
查看>>
让ul中的li居中显示
查看>>
区分super和this
查看>>
最近工作
查看>>
XJOI网上同步训练DAY2 T2
查看>>
Codeforces 509F Progress Monitoring
查看>>
spring cloud: eureka搭建
查看>>
导弹拦截
查看>>
两个被广泛使用的Model Checking工具
查看>>
BZOJ 4999 This Problem Is Too Simple!
查看>>
[HDU]3555Bomb
查看>>