当前位置: 首页 > backend >正文

LeetCode 3341到达最后一个房间的最少时间 I 题解

这题是一道比较典型的bfs,但是题目会在此基础上加一些限制,先给出示例和图示让大家更容易理解这题的思路

输入:moveTime = [[0,4],[4,4]]输出:6解释:需要花费的最少时间为 6 秒。- 在时刻 t == 4,从房间 (0, 0) 移动到房间 (1, 0) ,花费 1 秒。
- 在时刻 t == 5 ,从房间 `(1, 0)` 移动到房间 (1, 1) ,花费 1 秒。

典型的bfs只是要边bfs的时候边判断t是需要wait后加1还是直接加1,对于此题我们不仅仅可以使用bfs也可以使用Dijkstra,Dijkstra的逻辑感可能更强些,但是会用到一个我平常不怎么使用的函数式子。

先来看看Dijkstra的示例实现,分步说说刚好巩固自身对于该方法的使用,因为我也不太熟悉。

先建立上、下、左、右的走步坐标系

int[][] dirs = {{-1, 0}, {1, 0}, {0, -1}, {0, 1}};

然后建立队列存放数据,但这个队列有些不同,它不是先进先出(FIFO)类型的队列,而是根据数组第一个值的大小决定,最小的值先出栈

PriorityQueue<int[]> heap = new PriorityQueue<>((a, b) -> a[0] - b[0]);

这里解释很简单,就是初始化一下坐标的开始位置{t,x,y},t表示抵达(x,y)位置的时间,将这三个值一起记录下来方便遍历比较

heap.offer(new int[]{0, 0, 0});

这里的话算是一种打标记,记录每个位置的抵达时间,如果有比初始的值小的就替换然后进栈遍历后续节点

int n = moveTime.length, m = moveTime[0].length;
// time[i][j]表示到达(i,j)的最少时间
int[][] time = new int[n][m];
for (int i = 0; i < n; i++) {Arrays.fill(time[i], Integer.MAX_VALUE);
}
time[0][0] = 0; // 初始化

这里就是Dijkstra最关键的一个部分,先来解释一下这里的代码,将初始节点和后续节点抛出(选取最小的t节点)对于t>time[x][y]的值是因为之前遍历过了(x,y)这个节点且其对应的抵达时间相较于当前的t更小,所以不需要继续往下了
举个例子:
假设当前到 (1,1) 有两条路径:

  • 一条需要时间 10,先入队。

  • 后来发现另一条更优路径,时间为 6,也入队。

此时队列中两个 (1,1)

[ (6, 1, 1), (10, 1, 1) ]
  • 6 的那条先处理,更新时间为 6

  • 再处理 10 的那条,因为你之前已经更新为 6,所以直接跳过。

    if (t > time[x][y]) continue; // 剪枝

如果不剪枝会怎么样?

  • 会再次访问 (1,1),重复处理邻居,导致不必要的入队操作,性能变差。
while (!heap.isEmpty()) {int[] curr = heap.poll();int t = curr[0], x = curr[1], y = curr[2];if (t > time[x][y]) { // 剪枝continue;}for (int[] dir : dirs) {int nx = x + dir[0], ny = y + dir[1];if (0 <= nx && nx < n && 0 <= ny && ny < m) {int nt;if (t < moveTime[nx][ny]) { // 需要等待nt = 1 + moveTime[nx][ny];} else { // 否则,直接进入nt = t + 1;}if (nt < time[nx][ny]) { // 当前的更优路径time[nx][ny] = nt;heap.offer(new int[]{nt, nx, ny});}}}}

之后就是进入bfs的遍历进行判断当前节点向四周延申的节点是需要等待后才能进入还是可以直接进入,并更新路径time[nx][ny],以便后续的节点遍历至相同节点进行对比抵达时间然后更新。

最后return time[n-1][m-1]即可

以下是完整代码

class Solution {public int minTimeToReach(int[][] moveTime) {// Dijkstra,并允许同一个节点的重复访问int[][] dirs = {{-1, 0}, {1, 0}, {0, -1}, {0, 1}};// 最小堆,存储数组 {t, x, y},表示到达 (x, y) 的时间为 tPriorityQueue<int[]> heap = new PriorityQueue<>((a, b) -> a[0] - b[0]);// 起点heap.offer(new int[]{0, 0, 0});int n = moveTime.length, m = moveTime[0].length;// time[i][j]表示到达(i,j)的最少时间int[][] time = new int[n][m];for (int i = 0; i < n; i++) {Arrays.fill(time[i], Integer.MAX_VALUE);}time[0][0] = 0; // 初始化while (!heap.isEmpty()) {int[] curr = heap.poll();int t = curr[0], x = curr[1], y = curr[2];if (t > time[x][y]) { // 剪枝continue;}for (int[] dir : dirs) {int nx = x + dir[0], ny = y + dir[1];if (0 <= nx && nx < n && 0 <= ny && ny < m) {int nt;if (t < moveTime[nx][ny]) { // 需要等待nt = 1 + moveTime[nx][ny];} else { // 否则,直接进入nt = t + 1;}if (nt < time[nx][ny]) { // 当前的更优路径time[nx][ny] = nt;heap.offer(new int[]{nt, nx, ny});}}}}return time[n - 1][m - 1]; // 终点}
}

这是用了方法优先队列对此题的优解,也可以使用传统的bfs队列,只是会时间用的更长一些

class Solution {public int minTimeToReach(int[][] moveTime) {int n = moveTime.length, m = moveTime[0].length;int[][] dirs = {{0, 1}, {1, 0}, {0, -1}, {-1, 0}};// 初始化距离矩阵int[][] dist = new int[n][m];for (int i = 0; i < n; i++) {Arrays.fill(dist[i], Integer.MAX_VALUE);}dist[0][0] = 0;// 使用双端队列进行BFSDeque<int[]> q = new ArrayDeque<>();q.offer(new int[]{0, 0});while (!q.isEmpty()) {int[] curr = q.poll();int i = curr[0], j = curr[1];for (int[] dir : dirs) {int ni = i + dir[0], nj = j + dir[1];if (ni >= 0 && ni < n && nj >= 0 && nj < m) {// 计算从(i,j)到达(ni,nj)的最早时间int newTime = Math.max(dist[i][j], moveTime[ni][nj]) + 1;// 如果找到更优路径,更新并加入队列if (newTime < dist[ni][nj]) {dist[ni][nj] = newTime;q.offer(new int[]{ni, nj});}}}}return dist[n - 1][m - 1];}
}

也一样需要一个二维数组去存抵达时间的值,然后更新最小的抵达时间,这个应该比上一个更容易理解,不想再解释了…

http://www.xdnf.cn/news/4744.html

相关文章:

  • 基于大模型的计划性剖宫产全流程预测与方案优化研究报告
  • 跨浏览器自动化测试的智能生成方法
  • rom定制系列------红米note12 5G版miui14修改型号root版 原生安卓14批量线刷固件 原生安卓15等
  • STM32 ADC
  • 可撤销并查集,原理分析,题目练习
  • 数据结构(三)——栈和队列
  • 《P2880 [USACO07JAN] 平衡系列 G》
  • 【基础复习笔记】计算机视觉
  • 笔记本电脑实现网线内网 + Wi-Fi外网同时使用的配置方案
  • 运维打铁:服务器分类及PHP入门
  • 移植easylogger通过J-Linker的RTT输出日志/Ozone的RTT设置
  • 华为设备MSTP
  • 【IP101】图像压缩技术详解:从JPEG到小波压缩的完整指南
  • 机器人领域和心理学领域 恐怖谷 是什么
  • 如何为APP应用程序选择合适的服务器
  • C++ - 输入输出
  • Matlab 车辆四自由度垂向模型平稳性
  • Jupyter Notebook / Lab 疑难杂症记:从命令找不到到环境冲突与网络阻塞的排查实录
  • 手撕基于AMQP协议的简易消息队列-8(单元测试的编写)
  • linux mutex 互斥锁实现
  • 【网工第6版】第7章 网络操作系统与应用服务器③
  • 芯片测试之Open-Short Test全解析:从原理到实战
  • SpringBoot教程(vuepress版)
  • AWS VPC架构师指南:从零设计企业级云网络隔离方案
  • C语言if语句的用法(非常详细,通俗易懂)
  • CentOS7将yum源更换为阿里源
  • 2025年通信安全员考试题库及答案
  • 【Linux系统】第三节—权限
  • 线索二叉树
  • Arm核的Ubuntu系统上安装Qt