【算法】BFS算法解决多源最短路问题(C++)

文章目录

  • 前言
    • 那么什么是单源最短路 / 多源最短路呢?
    • 如何解决此类题?
      • 解法一
      • 解法二
        • 对于解法二,如何编写代码?
  • 算法题
    • 542.01矩阵
    • 1020.飞地的数量
    • 1765.地图中的最高点
    • 1162.地图分析

前言

此前我们对 单源最短路 问题进行的讲解:

使用bfs算法解决单源最短路问题


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

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

在这里插入图片描述

而对于多源最短路问题:

在这里插入图片描述


如何解决此类题?

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

解法一

在这里插入图片描述

解法二

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

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

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

我们对于 单源最短路 问题的bfs解法为:

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

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

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

在这里插入图片描述

算法题

542.01矩阵

在这里插入图片描述

思路

  • 题意分析:题目要求返回一个数组,该数组的每一位都是矩阵中当前位置到最近的0的距离。这个数组我们叫做dist数组,下面多道题都会用上,
  • 解法一 对每个位置都进行扩展
    在这里插入图片描述
  • 解法二: bfs + 正难则反 + dist数组
    • 正难则反:将思维逆转!我们以0作为起点,对矩阵进行扩容
    • dist数组:也是结果数组,开始初始化为-1,对于dist[i][j]:
      • dist[i][j] == -1:未被检索过
      • dist[i][j] != -1:该位置到0的最短距离
    1. 将矩阵的所有起始点0的位置加入矩阵
    2. 根据下图的思路,每次对队列中的元素进行扩展
      在这里插入图片描述

代码

class Solution {
public:
    int dx[4] = {0, 0, -1, 1};
    int dy[4] = {-1, 1, 0 ,0};
    vector<vector<int>> updateMatrix(vector<vector<int>>& mat) {
        int m = mat.size(), n = mat[0].size();
        vector<vector<int>> dist(m, vector<int>(n, -1)); // 将dist数组元素初始化为-1(未检索的状态)
        queue<pair<int, int>> q;
        // 正难则反:以矩阵中的0作为起点进行扩展
        // 将所有起始点入队
        for(int i = 0; i < m; ++i)
            for(int j = 0; j < n; ++j)
                if(mat[i][j] == 0)
                {
                    q.push({i, j});
                    dist[i][j] = 0;
                }

        // 对队列中的元素进行拓展
        while(q.size())
        {
            auto [a, b] = q.front();    q.pop();
            for(int i = 0; i < 4; ++i)
            {
                int x = dx[i] + a, y = dy[i] + b;
                if((x >= 0 && x < m) && (y >= 0 && y < n) && dist[x][y] == -1)
                {
                    dist[x][y] = dist[a][b] + 1;
                    q.push({x, y});
                }
            }
        }
        return dist;
    }
};

1020.飞地的数量

在这里插入图片描述

思路

  • 题意分析:题目要求返回无法离开水域的陆地单元格个数,即在边界内且被水域完全包裹的单元格。
  • 解法 bfs + 正难则反
    • 正难则反:我们以四边的1为起点,向内扩展
    • 多源bfs:将四边的1作为起点入队,将相连接的陆地单元格标记
    • 最后遍历visited数组,统计结果

代码

class Solution {
public:
    int dx[4] = {0, 0, -1, 1};
    int dy[4] = {-1, 1, 0, 0};

    int numEnclaves(vector<vector<int>>& grid) {
        int m = grid.size(), n = grid[0].size();
        // 将visited数组元素初始化为0
        vector<vector<bool>> visited(m, vector<bool>(n, false));
        queue<pair<int, int>> q;
        // 正难则反:以边界的1为起始点,进行bfs
        // 1. 找到边界1,并扩展
        for(int i = 0; i < m; ++i)
            for(int j = 0; j < n; ++j)
                if(((i == 0 || i == m-1) || (j == 0 || j == n-1)) && grid[i][j] == 1)
                {
                    q.push({i, j});
                    visited[i][j] = true;
                }

        // 2. 多源bfs
        while(q.size())
        {
            auto [a, b] = q.front(); q.pop();
            for(int i = 0; i < 4; ++i)
            {
                int x = a + dx[i], y = b + dy[i];
                if((x >= 0 && x < m) && (y >= 0 && y < n) && !visited[x][y] && grid[x][y])
                {
                    q.push({x, y});
                    visited[x][y] = true;
                }
            }
        }

        // 3. 提取结果
        int ret = 0;
        for(int i = 0; i < m; ++i)
            for(int j = 0; j < n; ++j)
                if(!visited[i][j] && grid[i][j])
                    ++ret;

        return ret;
    }
};

1765.地图中的最高点

在这里插入图片描述

思路

  • 题意分析:题目给出一个矩阵,1为水域,0为陆地,要求我们自行安排陆地高度,使矩阵的最高高度最大(相邻单元格高度差 <= 1)
  • 解法 bfs + dist数组
  • 思路
    1. 首先创建dist数组,原理与第一题01矩阵一样,将矩阵中所有水域的位置在dist中初始化为0,并插入到队列中。
    2. 由于要求矩阵的最高高度最大,我们在设计高度时自然要将高度差都设为最大值,即1
    3. 随后进行多源bfs的操作,综上我们发现,这道题理清思路后,代码方面和“01矩阵”完全一致

代码

class Solution {
public:
    int dx[4] = {0, 0, -1, 1};
    int dy[4] = {-1, 1, 0, 0};

    vector<vector<int>> highestPeak(vector<vector<int>>& isWater) {
        // 思路同 “10矩阵”
        int m = isWater.size(), n = isWater[0].size();
        vector<vector<int>> dist(m, vector<int>(n, -1));
        queue<pair<int, int>> q;

        // 1.对水域进行扩展
        for(int i = 0; i < m; ++i)
            for(int j = 0; j < n; ++j)
                if(isWater[i][j] == 1) // 1是水域
                {
                    q.push({i, j});
                    dist[i][j] = 0;
                }

        // 2.
        while(q.size())
        {
            auto [a, b] = q.front();    q.pop();
            for(int i = 0; i < 4; ++i)
            {
                int x = a + dx[i], y = b + dy[i];
                if((x >= 0 && x < m) && (y >= 0 && y < n) && dist[x][y] == -1)
                {
                    q.push({x, y});
                    dist[x][y] = dist[a][b] + 1;
                }
            }
        }

        return dist;
    }
};

1162.地图分析

在这里插入图片描述

思路

  • 题意分析:题目要求找到离陆地最远的海洋单元格的曼哈顿距离, 曼哈顿距离 简单解释为:
    在这里插入图片描述

  • 解法思路 bfs + 正难则反 + dist数组

    • 正难则反:题目要求找海洋到陆地的最大距离,我们以陆地单元格作为起始点,向内扩展
    • 同样的思路,我们首先遍历数组,将陆地单元格位置存入到队列中,并标记dist的位置
    • 后执行多源bfs的代码思路即可

代码

class Solution {
public:
    int dx[4] = {0, 0, -1, 1};
    int dy[4] = {-1, 1, 0, 0};
    int maxDistance(vector<vector<int>>& grid) {
        int m = grid.size(), n = grid[0].size();
        // 正难则反:以1为起点,向内扩展并标记
        vector<vector<int>> dist(m, vector<int>(n, -1));
        queue<pair<int, int>> q;
        // 1. 根据单元格1向内扩展
        for(int i = 0; i < m; ++i)
            for(int j = 0; j < n; ++j)
                if(grid[i][j] == 1)
                {
                    q.push({i, j});
                    dist[i][j] = 0;
                }

        // 2. 多源bfs
        int ret = -1; // 统计结果
        while(q.size())
        {
            auto [a, b] = q.front();    q.pop();
            for(int i = 0; i < 4; ++i)
            {
                int x = a + dx[i], y = b + dy[i];
                if((x >= 0 && x < m) && (y >= 0 && y < n) && dist[x][y] == -1)
                {
                    q.push({x, y});
                    dist[x][y] = dist[a][b] + 1;
                    ret = max(ret, dist[x][y]);
                }
            }
        }

        return ret;
    }
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/355709.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Linux中并发程序设计(进程的创建和回收、exec函数使用、守护进程创建和使用、GDB的父、子进程代码的调试、线程的创建和参数传递)

进程的创建和回收 进程概念 概念 程序 存放在磁盘上的指令和数据的有序集合&#xff08;文件&#xff09; 静态的 进程 执行一个程序所分配的资源的总称 动态的进程和程序比较 注&#xff1a;进程是存在RAM中&#xff0c;程序是存放在ROM(flash)中的进程内容 BSS段&#xff…

一篇文章搞懂Python的文件读写

目录 一、打开文件 二、读取文件内容 三、写入文件内容 四、追加内容到文件末尾 五、文件路径 六、错误处理 七、关闭文件 八、文件读写的高级功能 总结 在Python中&#xff0c;文件读写是一项常见的操作。通过文件读写&#xff0c;我们可以保存数据&#xff0c;加载数…

算法学习记录:动态规划

前言&#xff1a; 算法学习记录不是算法介绍&#xff0c;本文记录的是从零开始的学习过程&#xff08;见到的例题&#xff0c;代码的理解……&#xff09;&#xff0c;所有内容按学习顺序更新&#xff0c;而且不保证正确&#xff0c;如有错误&#xff0c;请帮助指出。 学习工具…

Find My资讯|苹果Apple Pencil 3将支持Find My定位查找功能

国外科技媒体 9to5Mac 深挖 iOS 17.4 Beta 1 更新代码&#xff0c;发现了 Apple Pencil 3 的踪迹&#xff0c;并显示该手写笔支持 Find My 功能。这就意味着可以和 AirPods 和 AirTags 一样&#xff0c;用户可以通过 Find My 网络&#xff0c;寻找丢失的 Apple Pencil 手写笔。…

Kotlin for loop: in、 until、 step、 downTo

Kotlin for loop: in、 until、 step、 downTo fun loop1() {for (i in 0..5) {print("$i ")}println("\n1-end\n") }fun loop2() {for (i in 0 until 5) {print("$i ")}println("\n2-end\n") }fun loop3() {for (i in 0 until (5)) {…

探索Viper-适用于GoLang的完整配置解决方案

前言 对于现代应用程序&#xff0c;尤其大中型的项目来说&#xff0c;在程序启动和运行时&#xff0c;往往需要传入许多参数来控制程序的行为&#xff0c;我们可以通过命令行参数&#xff0c;环境变量&#xff0c;配置文件等方式来将参数传递给程序。而Viper库为Golang语言开发…

LabVIEW继电器触点接触电阻自动测试

继电器作为工业中的重要组件&#xff0c;其性能直接影响着整个生产线的可靠性和安全性。触点接触电阻是衡量继电器性能的重要参数&#xff0c;传统的测试方法效率低下且成本高昂。为了解决这些问题&#xff0c;采用LabVIEW软件&#xff0c;结合专业的硬件平台&#xff0c;实现了…

备战蓝桥杯----数据结构及STL应用(基础2)

上次我们讲了vector的大致内容&#xff0c;接下来让我们讲一下栈&#xff0c;队列吧&#xff01; 什么是栈呢&#xff1f; 很简单&#xff0c;我们用的羽毛球桶就是&#xff0c;我们取的球&#xff0c;是最后放的&#xff0c;栈是一种先进后出的数据结构。 方法函数 s.push(…

怎样做好Code Review

Code Review方案 定义 Code Review代码评审是指在软件开发过程中&#xff0c;通过对源代码进行系统性检查的过程。通常的目的是查找各种缺陷&#xff0c;包括代码缺陷、功能实现问题、编码合理性、性能优化等&#xff1b;保证软件总体质量和提高开发者自身水平 code review …

SRM是什么意思?SRM供应商管理系统哪个好?

阅读本文&#xff0c;您将了解&#xff1a;一、SRM是什么意思&#xff1b;二、SRM的应用价值&#xff1b;三、SRM供应商管理系统哪个好&#xff1f;推荐在零代码平台自主开发。 在当今快速变化的商业环境中&#xff0c;企业之间的协作和效率至关重要。供应链管理作为企业运营的…

HarmonyOS鸿蒙学习笔记(24)AppScope的icon和UIAblity的icon的区别

在HarmonyOS中有两个配置文件app.json5配置文件和module.json5配置文件&#xff0c;两个配置文件都有icon和label标签&#xff1a; 二者区别如下&#xff1a; app.json5的icon app.json5里面的icon和label:对应的是应用图标和应用标签&#xff0c;应用图标和标签是在设置应用…

HarmonyOS --@state状态装饰器

在声明式UI中&#xff0c;是以状态驱动视图更新。 状态&#xff08;state&#xff09;&#xff1a;指驱动视图更新的数据&#xff08;被装饰器标记的变量&#xff09;。 试图&#xff08;view&#xff09;&#xff1a;基于UI描述渲染得到用户界面 State装饰器标记的变量必须初…

C# Socket 允许控制台应用通过防火墙

需求&#xff1a; 在代码中将exe添加到防火墙规则中&#xff0c;允许Socket通过 添加库引用 效果&#xff1a; 一键三联 若可用记得点赞评论收藏哦&#xff0c;你的支持就是写作的动力。 源地址: https://gist.github.com/cstrahan/513804 调用代码: private static void …

雅特力AT32 Workbench图形化代码生成工具,简化嵌入式开发利器

嵌入式系统应用市场广泛&#xff0c;早已遍及日常生活&#xff0c;随着产品需求复杂度的提升&#xff0c;32位MCU开发难度也随之增加&#xff0c;如何降低开发成本&#xff0c;缩短开发周期&#xff0c;是所有嵌入式开发人员的共同课题。 面对市场竞争日益加剧的情形&#xff…

【Web前端实操17】导航栏效果——滑动门

滑动门 定义: 类似于这种: 滑到导航栏的某一项就会出现相应的画面,里面有对应的画面出现。 箭头图标操作和引用: 像一些图标,如果需要的话,可以找字体图标,比如阿里巴巴矢量图标库:iconfont-阿里巴巴矢量图标库 选择一个——>添加至购物车——>下载代码 因…

实战 | OpenCV+OCR实现弧形文字识别实例(详细步骤 + 源码)

导 读 本文主要介绍基于OpenCV+OCR实现弧形文字识别实例,并给详细步骤和代码。源码在文末。 背景介绍 测试图如下,目标是正确识别图中的字符。图片来源: https://www.51halcon.com/forum.php?mod=viewthread&tid=6712 同样,论坛中已经给出了Halcon实现代码,…

Linux文件管理(下)

上上篇介绍了Linux文件管理的上部分内容&#xff0c;这次继续将 Linux文件管理的剩余部分说完。内容如下。 一、查看文件内容 1、cat 命令 1.1 输出文件内容 基本语法&#xff1a; cat 文件名称主要功能&#xff1a;正序输出文件的内容。 eg&#xff1a;输出 readme.txt文…

《幻兽帕鲁》1月29日游戏服务器推荐!腾讯云降低规格再次降价!

腾讯29日刷新规格&#xff0c;从14M降低到12M&#xff0c;硬盘和流量都有降低&#xff0c;但价格打下来了&#xff01;价格从66元/月降低到32元/月&#xff0c;277元/3个月降低到96元/3个月&#xff01; 三大厂商4核16G的云服务器价格对齐&#xff0c;不过具体参数略有不同 阿里…

w24文件上传之PHP伪协议

PHP支持的伪协议 file:// - 访问本地文件系统 http:// - 访问网址 ftp:// - 访问文件 php:// -访问各个输入/输出流 zlib:// -压缩流 data:// - 数据 glob:// -查找匹配的文件路径模式 phar:// - php归档 ssh2:// - Secure shell 2 rar:// - RAR ogg:// - 音频流 expect:// - …

力扣(LeetCode)227. 基本计算器 II

给你一个字符串表达式 s &#xff0c;请你实现一个基本计算器来计算并返回它的值。 整数除法仅保留整数部分。 你可以假设给定的表达式总是有效的。所有中间结果将在 [-231, 231 - 1] 的范围内。 注意&#xff1a;不允许使用任何将字符串作为数学表达式计算的内置函数&#…