• 每日OJ题_多源BFS①_力扣542. 01 矩阵(多源BFS解决最短路原理)


    目录

    多源BFS解决最短路算法原理

    力扣542. 01 矩阵

    解析代码


    多源BFS解决最短路算法原理

    什么是单源最短路 / 多源最短路?

    之前的BFS解决最短路都是解决的单源最短路。

    画图来说,单源最短路问题即为:

    而对于多源最短路问题:

    如何解决此类题?

    自然是利用多源BFS解决,下面提出解法:

            当我们将所有的源点作为一个源点来进行解题时,问题又变成了单源最短路问题,而为什么可以认为这种解法是正确的呢?

    • 感性的理解 :对于上图的ABC三点,显然A点到目标点的距离更远,当我们将其作为一个点时,A点就会被直接排除,此时该特殊源点实际上就是最近的源点的合并。

    对于解法二,如何编写代码?

    对于 单源最短路 问题的BFS解法为:

    • 将起始点加入到队列中,再进行一层一层的扩展

    自然,对于 多源最短路 的BFS解法为:

    • 将所有的起始点加入到队列中,再进行一层一层的扩展

    力扣542. 01 矩阵

    542. 01 矩阵

    难度 中等

    给定一个由 0 和 1 组成的矩阵 mat ,请输出一个大小相同的矩阵,其中每一个格子是 mat 中对应位置元素到最近的 0 的距离。

    两个相邻元素间的距离为 1 。

    示例 1:

    输入:mat = [[0,0,0],[0,1,0],[0,0,0]]
    输出:[[0,0,0],[0,1,0],[0,0,0]]
    

    示例 2:

    输入:mat = [[0,0,0],[0,1,0],[1,1,1]]
    输出:[[0,0,0],[0,1,0],[1,2,1]]
    

    提示:

    • m == mat.length
    • n == mat[i].length
    • 1 <= m, n <= 10^4
    • 1 <= m * n <= 10^4
    • mat[i][j] is either 0 or 1.
    • mat 中至少有一个 
    1. class Solution {
    2. public:
    3. vectorint>> updateMatrix(vectorint>>& mat) {
    4. }
    5. };

    解析代码

    对于求的最终结果,有两种方式:

    • 第一种方式:从每一个 1 开始,然后通过层序遍历找到离它最近的 0 。这一种方式,会以所有的 1 起点,来一次层序遍历,势必会遍历到很多重复的点。并且如果矩阵中只有一个 0 的话,每一次层序遍历都要遍历很多层,时间复杂度较高。
    • 第二种方式:正难则反,从 0 开始层序遍历,并且记录遍历的层数。当第一次碰到 1 的时候,当前的层数就是这个 1 离 0 的最短距离。

            第二种方式,在遍历的时候标记一下处理过的 1 ,能够做到只用历整个矩阵一次,就能得到最终结果。 但是有一个问题, 0 是有很多个的,怎么才能保证遇到的 1 距离这一个 0 是最近的呢?可以先把所有的 0 都放在队列中,把它们当成一个整体,每次把当前队列里面的所有元素向外扩展一次。可以开一个dist数组就能完成类似前面BFS解决最短路所需的bool数组,step和size变量:初始化成-1的话就是没遍历的,遍历的step只需在前一个格子加1,层数也能确定。

    1. class Solution {
    2. int dx[4] = {0, 0, -1, 1};
    3. int dy[4] = {1, -1, 0, 0};
    4. public:
    5. vectorint>> updateMatrix(vectorint>>& mat) {
    6. int m = mat.size(), n = mat[0].size();
    7. vectorint>> dist(m, vector<int>(n, -1));
    8. queueint, int>> q;
    9. for(int i = 0; i < m; i++) // 把所有的源点加⼊到队列中
    10. {
    11. for(int j = 0; j < n; j++)
    12. {
    13. if(mat[i][j] == 0)
    14. {
    15. q.push({i, j});
    16. dist[i][j] = 0;
    17. }
    18. }
    19. }
    20. while(!q.empty()) // ⼀层⼀层往外扩
    21. {
    22. auto [a, b] = q.front();
    23. q.pop();
    24. for(int i = 0; i < 4; ++i)
    25. {
    26. int x = a + dx[i], y = b + dy[i];
    27. if(x >= 0 && x < m && y >= 0 && y < n && dist[x][y] == -1)
    28. {
    29. dist[x][y] = dist[a][b] + 1;
    30. q.push({x, y});
    31. }
    32. }
    33. }
    34. return dist;
    35. }
    36. };

    也可以不开空间直接在原数组操作

    1. class Solution {
    2. int dx[4] = {0, 0, -1, 1};
    3. int dy[4] = {1, -1, 0, 0};
    4. public:
    5. vectorint>> updateMatrix(vectorint>>& mat) {
    6. int m = mat.size(), n = mat[0].size();
    7. queueint, int>> q;
    8. for(int i = 0; i < m; i++) // 把所有的源点加⼊到队列中
    9. {
    10. for(int j = 0; j < n; j++)
    11. {
    12. if(mat[i][j] == 0)
    13. q.push({i, j});
    14. else
    15. mat[i][j] = -1;
    16. }
    17. }
    18. while(!q.empty()) // ⼀层⼀层往外扩
    19. {
    20. auto [a, b] = q.front();
    21. q.pop();
    22. for(int i = 0; i < 4; ++i)
    23. {
    24. int x = a + dx[i], y = b + dy[i];
    25. if(x >= 0 && x < m && y >= 0 && y < n && mat[x][y] == -1)
    26. {
    27. mat[x][y] = mat[a][b] + 1;
    28. q.push({x, y});
    29. }
    30. }
    31. }
    32. return mat;
    33. }
    34. };

  • 相关阅读:
    gradle-3-(configure篇)
    TypeScript深度掌握
    P44,45 属性预处理,执行后游戏效果回调,附录指定区域内修改变量
    x汽车登陆网站登陆rsa加密逆向
    深入理解Python虚拟机:super超级魔法的背后原理
    flutter GetMaterialAPP unknownRoute 失效
    Python编程-- BS4解析
    python类中的下划线
    计算机操作系统第四版第八章磁盘存储器的管理—课后习题答案
    创意中秋与国庆贺卡 - 用代码为节日增添喜悦
  • 原文地址:https://blog.csdn.net/GRrtx/article/details/136938083