一、摘要

本文介绍了一种使用BFS求无向图中第二短的距离的算法,本文算法参考自Python3 BFS找出第二短路径。通过使用BFS求出题目中给出的路径图(无向联通图)从节点1到节点n的第二短的路径,再根据路径长度计算出需要耗费的时间。其实在LeetCode官网本题目的题解栏目下已经有很多优秀的题解了,本文除了是使用C++与其他题解有些许不同外无任何优势。还是建议读者阅读其他高赞题解。

二、题解

由于给给节点上红灯、绿灯的亮暗是一致的,因此将各节点间的边长视作1,只要求出从节点0节点n-1经过的路径长度即可得到总的耗费时间。

因此本题就变成了求节点0节点n-1第二短的长度

需要注意的是,我们要求的是绝对的第二短的路径,如果有两条最短的不同路径长度都为x,那么我还需要继续求得一个大于x的第二短路径。既然是等边长图中,求最短路径我们可以使用BFS。

使用一个队列queue<pair<int, int>>记录所搜到的节点id和从节点0到该节点id的耗时

为了求第二短的距离因此我们是用两个数组vector<int> firstvector<int> second分别记录节点0节点id的第一、二短耗时(初始时都设为最大值,在C++中即为INT_MAX)。

在使用队列进行bfs算法时:

  • 如果下一个可能要入队列的节点n_id需要的耗时n_time小于first[n_id],那么说明节点0节点n_id的最短耗时为n_time,需要入队pair<int,int>(n_id, n_time)``。
  • 如果下一个可能要入队列的节点n_id需要的耗时n_time等于first[n_id],那么说明存在重复的 从节点0节点n_id的耗时相同、但路径不同 的最短耗时,此时不需要入队;
  • 如果下一个可能要入队列的节点n_id需要的耗时n_time大于first[n_id],且小于second[n_id],那么说明节点0节点n_id的第二短耗时为n_time,需要入队pair<int, int>(n_id, n_time)
  • 如果下一个可能要入队列的节点n_id需要的耗时n_time大于等于second[n_id],那么说明存在重复的 从节点0节点n_id的耗时相同、但路径不同 的第二短耗时、或者这是第三短的耗时,此时不需要入队;

代码如下:

class Solution {
public:
// 图的节点
struct Node{
int idx; // 节点id
vector<int> neighbor; // 相邻的节点id
Node(int i):idx(i){}
}; // 根据此时的时间now计算经过一条边,到达下一个节点的时间
int getNextTime(int now, int time, int change){
if((now/change)%2==0){
now += time;
}else{
now += (change - now%change)+time;
}
return now;
}
int secondMinimum(int n, vector<vector<int>>& edges, int time, int change) {
nodes_.resize(n);
for(int i=0; i<n; i++){
nodes_[i] = new Node(i);
}
// 建图
for(int i=0; i<edges.size(); i++){
int a = edges[i][0] - 1;
int b = edges[i][1] - 1;
nodes_[a]->neighbor.push_back(b);
nodes_[b]->neighbor.push_back(a);
} queue<pair<int, int>> que;
vector<int> first(n, INT_MAX); // first记录到达节点i最短的耗时
vector<int> second(n, INT_MAX); // second记录到达节点i第二短的耗时 que.push(pair<int, int>(0,0)); // 将(0,0)入队,表示达到节点0的耗时为0 while(que.empty()==false){
int id = que.front().first;
int now = que.front().second;
que.pop();
for(int i=0; i<nodes_[id]->neighbor.size(); i++){
int n_id = nodes_[id]->neighbor[i];
int n_time = getNextTime(now, time, change);
if(n_id==n-1){ // 如果是第n个节点
if(first[n_id]==INT_MAX){ // 第一次到达节点n-1,更新first,且pair<int,int>(n_id,n_time)入队
first[n_id] = n_time;
que.push(pair<int, int>(n_id, n_time));
}else if(n_time>first[n_id] && second[n_id]==INT_MAX){ // 到达节点n-1的第二短距离,返回答案
second[n_id] = n_time;
return n_time;
}
}else{
if(first[n_id]==INT_MAX){ // 第一次到达节点n_id,更新first,且pair<int,int>(n_id,n_time)入队
first[n_id] = n_time;
que.push(pair<int, int>(n_id, n_time));
}else if(n_time>first[n_id] && second[n_id]==INT_MAX){ // 到达节点n-1的第二短的距离,更新second,且pair<int,int>(n_id,n_time)入队
second[n_id] = n_time;
que.push(pair<int, int>(n_id, n_time));
}
}
}
}
return 0;
}
vector<Node*> nodes_;
};

复杂度分析:

  • 时间复杂度:O(n+m)
  • 空间复杂度:O(n+m)

    其中n为节点个数,m为边个数

[LeetCode] 2045. 到达目的地的第二短时间的更多相关文章

  1. 构造定律(constructal law)-构造定律作为第二个时间箭头,将和热力学第二定律一道将宇宙推向无序。

    优化系统结构,使信息和物质流在结构内的流动更畅通. 构造定律(constructal law) 由Adrian Bejan于1995创立的构造定律(constructal law):   For a ...

  2. Leetcode春季打卡活动 第二题:206. 反转链表

    Leetcode春季打卡活动 第二题:206. 反转链表 206. 反转链表 Talk is cheap . Show me the code . /** * Definition for singl ...

  3. 广搜最短路(最短时间到达目的地),POJ(3669)

    题目链接:http://poj.org/problem?id=3669 解题报告: 1.流星坠落的点,四周和自己本身都被毁灭,不断更新每个点被毁灭的时候的最短时间. 2.搜索终点是,到达某个点,这个不 ...

  4. [LeetCode] Next Closest Time 下一个最近时间点

    Given a time represented in the format "HH:MM", form the next closest time by reusing the ...

  5. 动态规划之DP中判断是否到达某一状态(最短时间是什么)?

    codevs1684 垃圾陷阱  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 黄金 Gold   题目描述 Description 卡门——农夫约翰极其珍视的一条Holste ...

  6. [LeetCode] Employee Free Time 职员的空闲时间

    We are given a list schedule of employees, which represents the working time for each employee. Each ...

  7. LeetCode 176. Second Highest Salary (第二高的薪水)

    题目标签: 题目给了我们一个工资表,让我们返回第二高的工资. 利用Max,把第一高的工资找到,然后利用 NOT IN,去找到第二高的工资. Java Solution: Runtime:  153ms ...

  8. leetcode.1266访问所有点的最小时间

    平面上有 n 个点,点的位置用整数坐标表示 points[i] = [xi, yi].请你计算访问所有这些点需要的最小时间(以秒为单位). 你可以按照下面的规则在平面上移动: 每一秒沿水平或者竖直方向 ...

  9. Java实现 LeetCode 754 到达终点数字(暴力+反向)

    754. 到达终点数字 在一根无限长的数轴上,你站在0的位置.终点在target的位置. 每次你可以选择向左或向右移动.第 n 次移动(从 1 开始),可以走 n 步. 返回到达终点需要的最小移动次数 ...

  10. 第二章 时间控件(DateTime Picker)

    这家伙太懒了,碰到问题才写博文,嘿嘿. 好了进入正题,二话不说,先放地址: 中文:http://www.bootcss.com/p/bootstrap-datetimepicker/index.htm ...

随机推荐

  1. 限流设置之Nginx篇

    question1:为什么用到Nginx,Nginx有什么功能? 1.反向代理(建议先看正向代理,反向代理则是同样你要与对方服务器建立连接,但是,代理服务器和目标服务器在一个LAN下,所以我们需要与代 ...

  2. 洛谷P1102 过河卒

    P1102 过河卒 链接在此 过河卒 此题如果直接忽略掉马的影响的话,可以看出很简单的递推规律 即 \[dp[i][j]=dp[i-1][j]+dp[i][]j-1] \] 也就是说,由于卒只能走直线 ...

  3. 正则表达式,js、javascript 的 replace 的坑,严重留意。

    一致以来我以为js的 replace 是全部替换的,没想到是只替换第一个,使用时要严重留意. 举例: let wokao: string = "abc + a_b_c + a.b.c&quo ...

  4. Python-目录下相同格式的Excel文件合并

    最近在客户现场接到一个任务,需要将全国所有省份的数据进行合并.目录是分层级的,首先是省份目录.然后地级市目录.最里面是区县目录.需要将每个目录中的数据进行合并,然后添加4列数据,并将某一个列的数据进行 ...

  5. JS leetcode 最长公共前缀 题解分析

    壹 ❀ 引 今天做的又是一道让我沮丧的题,思路有,但是代码逻辑最后还是没能正确理出来,题名为最长公共前缀,题目如下: 编写一个函数来查找字符串数组中的最长公共前缀. 如果不存在公共前缀,返回空字符串 ...

  6. NC20259 [SCOI2007]降雨量

    题目链接 题目 题目描述 我们常常会说这样的话:"X年是自Y年以来降雨量最多的".它的含义是X年的降雨量不超过Y年,且对于任意 Y<Z<X,Z年的降雨量严格小于X年. ...

  7. 实例详解在Go中构建流数据pipeline

    本文分享自华为云社区<Go并发范式 流水线和优雅退出 Pipeline 与 Cancellation>,作者:张俭. 介绍 Go 的并发原语可以轻松构建流数据管道,从而高效利用 I/O 和 ...

  8. 《系列二》-- 3、FactoryBean 的使用

    目录 FactoryBean 解决的问题 FactoryBean 接口初识 改造结果 最后的补充 回顾下 FactoryBean 的应用 factory-method 和 factory-bean 的 ...

  9. 新零售SaaS架构:什么是订单履约系统?

    什么是订单履约系统? 订单履约系统用来管理从接到销售订单,到把货品送到客户手中的整个业务过程.它是上游交易(如销售和客户下单环节)和下游仓储配送(如库存管理.物流)之间的桥梁,确保信息流的顺畅和操作的 ...

  10. FastGateway 发布v0.0.0.5

    FastGateway 发布v0.0.0.5 修复构建错误 修复docker-compose执行目录 修改请求来源分析数据列表展示 update README.md. 增加默认证书 修复构建脚本目录错 ...