6054. 逃离火灾

简介: 6054. 逃离火灾

说在前面

🎈每天进行一道算法题目练习,今天的题目是“逃离火灾”。

问题描述

给你一个下标从 0 开始大小为 m x n 的二维整数数组 grid ,它表示一个网格图。每个格子为下面 3 个值之一:

0 表示草地。

1 表示着火的格子。

2 表示一座墙,你跟火都不能通过这个格子。

一开始你在最左上角的格子 (0, 0) ,你想要到达最右下角的安全屋格子 (m - 1, n - 1) 。每一分钟,你可以移动到 相邻 的草地格子。每次你移动 之后 ,着火的格子会扩散到所有不是墙的 相邻 格子。

请你返回你在初始位置可以停留的 最多 分钟数,且停留完这段时间后你还能安全到达安全屋。如果无法实现,请你返回 -1 。如果不管你在初始位置停留多久,你 总是 能到达安全屋,请你返回 10^9 。

注意,如果你到达安全屋后,火马上到了安全屋,这视为你能够安全到达安全屋。

如果两个格子有共同边,那么它们为 相邻 格子。

示例 1:

输入:grid = [[0,2,0,0,0,0,0],[0,0,0,2,2,1,0],[0,2,0,0,1,2,0],[0,0,2,2,2,0,2],[0,0,0,0,0,0,0]]
输出:3
解释:上图展示了你在初始位置停留 3 分钟后的情形。
你仍然可以安全到达安全屋。
停留超过 3 分钟会让你无法安全到达安全屋。

示例 2:

输入:grid = [[0,0,0,0],[0,1,2,0],[0,2,0,0]]
输出:-1
解释:上图展示了你马上开始朝安全屋移动的情形。
火会蔓延到你可以移动的所有格子,所以无法安全到达安全屋。
所以返回 -1 。

示例 3:

输入:grid = [[0,0,0],[2,2,0],[1,2,0]]
输出:1000000000
解释:上图展示了初始网格图。
注意,由于火被墙围了起来,所以无论如何你都能安全到达安全屋。
所以返回 109 。

提示:

m == grid.length
n == grid[i].length
2 <= m, n <= 300
4 <= m * n <= 2 * 104
grid[i][j] 是 0 ,1 或者 2 。
grid[0][0] == grid[m - 1][n - 1] == 0

思路分析

信息提取

阅读完题目后,我们可以知道以下信息:

  • 格子中可能有3种情况
    0 表示草地。
    1 表示着火的格子。
    2 表示一座墙,你跟火都不能通过这个格子。
  • 起点和终点
    一开始你在最左上角的格子 (0, 0) ,你想要到达最右下角的安全屋格子 (m - 1, n - 1)
  • 火势会向四周扩散
    每秒中火势都会向四周扩散一格,但墙可以阻隔火势扩散。

解题步骤

  • 1、计算每一个格子火势最快蔓延到的时间
    我们可以使用dfs来模拟火势向四周扩散,并记录每一个个格子火势最快蔓延到的时间。
let fireMap = new Array(grid.length);
let map = new Array(grid.length);
for (let i = 0; i < fireMap.length; i++) {
    fireMap[i] = new Array(grid[0].length).fill(Infinity);
    map[i] = new Array(grid[0].length).fill(Infinity);
}
const fire = function (x, y, step) {
    if (x < 0 || y < 0 || x >= grid.length || y >= grid[0].length) return;
    if (grid[x][y] == 2 || fireMap[x][y] <= step) return;
    fireMap[x][y] = step;
    for (let i = 0; i < 4; i++) {
      fire(x + dx[i], y + dy[i], step + 1);
    }
};
for (let i = 0; i < grid.length; i++) {
    for (let j = 0; j < grid[i].length; j++) {
      if (grid[i][j] == 1) {
        fire(i, j, 0);
      }
    }
}
  • 2、计算在初始位置可以停留的 最多 分钟数
    我们可以使用dfs来搜索可以走到终点的路径,计算到达路径上格子所需步数和火势蔓延到该格子的时间只差,其中的最小值即为在初始位置可以停留的 最多 分钟数。
let res = -Infinity;
let dfs = function (x, y, step, min) {
    if (res == Infinity) return;
    if (x == grid.length - 1 && y == grid[x].length - 1) {
      min = Math.min(min, fireMap[x][y] - step);
      res = Math.max(res, min);
      return;
    }
    if (x < 0 || y < 0 || x >= grid.length || y >= grid[0].length) return;
    if (grid[x][y] == 2 || fireMap[x][y] <= step) return;
    if (map[x][y] <= step) return;
    map[x][y] = step;
    for (let i = 0; i < 4; i++) {
      if (res == Infinity) return;
      dfs(
        x + dx[i],
        y + dy[i],
        step + 1,
        Math.min(min, fireMap[x][y] - step - 1)
      );
    }
};
dfs(0, 0, 0, Infinity);

AC代码

/**
 * @param {number[][]} grid
 * @return {number}
 */
var maximumMinutes = function (grid) {
  const dx = [0, 1, 0, -1],
    dy = [1, 0, -1, 0];
  let fireMap = new Array(grid.length);
  let map = new Array(grid.length);
  for (let i = 0; i < fireMap.length; i++) {
    fireMap[i] = new Array(grid[0].length).fill(Infinity);
    map[i] = new Array(grid[0].length).fill(Infinity);
  }
  const fire = function (x, y, step) {
    if (x < 0 || y < 0 || x >= grid.length || y >= grid[0].length) return;
    if (grid[x][y] == 2 || fireMap[x][y] <= step) return;
    fireMap[x][y] = step;
    for (let i = 0; i < 4; i++) {
      fire(x + dx[i], y + dy[i], step + 1);
    }
  };
  for (let i = 0; i < grid.length; i++) {
    for (let j = 0; j < grid[i].length; j++) {
      if (grid[i][j] == 1) {
        fire(i, j, 0);
      }
    }
  }
  let res = -Infinity;
  let dfs = function (x, y, step, min) {
    if (res == Infinity) return;
    if (x == grid.length - 1 && y == grid[x].length - 1) {
      min = Math.min(min, fireMap[x][y] - step);
      res = Math.max(res, min);
      return;
    }
    if (x < 0 || y < 0 || x >= grid.length || y >= grid[0].length) return;
    if (grid[x][y] == 2 || fireMap[x][y] <= step) return;
    if (map[x][y] <= step) return;
    map[x][y] = step;
    for (let i = 0; i < 4; i++) {
      if (res == Infinity) return;
      dfs(
        x + dx[i],
        y + dy[i],
        step + 1,
        Math.min(min, fireMap[x][y] - step - 1)
      );
    }
  };
  dfs(0, 0, 0, Infinity);
  if (res == Infinity) return 1000000000;
  return res == -Infinity ? -1 : res;
};

说在后面

🎉这里是JYeontu,现在是一名前端工程师,有空会刷刷算法题,平时喜欢打打羽毛球🏸 ,平时也喜欢写些东西,既为自己记录📋,也希望可以对大家有那么一丢丢的帮助,写的不好望多多谅解🙇,写错的地方望指出,定会认真改进😊,在此谢谢大家的支持,我们下文再见🙌。

目录
相关文章
|
9月前
|
机器学习/深度学习
1191:流感传染
1191:流感传染
123 0
|
人工智能 安全
如果细菌病毒人人可以看得见
武汉新型病毒为什么那么可怕?首先,传播速度快,飞沫传播和接触传播,气溶胶和粪口等传播。其次,短期类没有预防的药物以及治疗药物,只能靠行为预防(勤洗手、戴口罩、不聚集、宅在家)。因此,给百姓带来恐慌,引发一系列的社会问题:防疫物资紧张,全民封闭在家,经济损失更是带数以亿万计。我就在想,人工智能能否让细菌病毒看得见,大家也就不那么恐慌,毕竟看得见的敌人比看不见的敌人要好得多。
585 0
珍爱生命,远离大论战
  前几天小学聚会,想起来了一些小时候的趣事。有一同学,他学习能力强,学东西快,老师讲的课程他都会了,作业也都写完了。然后呢呆着没事,找他同桌闲聊,放学了还拉着他疯玩。结果呢,第二天要交作业,他是没事了,可是他同桌惨了,忘记写作业了。
909 0