C++:bfs解决多源最短路与拓扑排序问题习题

news2025/1/20 6:39:12

1. 多源最短路

其实就是将所有源头都加入队列,

01矩阵

LCR 107. 01 矩阵 - 力扣(LeetCode)

 思路

求每个元素到离其最近0的距离如果我们将1当做源头加入队列的话,无法处理多个连续1的距离存储,我们反其道而行之,将所有的终点存进队列,遍历离其最近的1,当然如果旁边是0无需入队(因为旁边这个0找到的1肯定比他快一步),是1才要入队先遍历到1的步数一定是最小的步数

ac代码

class Solution {
public:

    vector<vector<int>> updateMatrix(vector<vector<int>>& mat) {
        int m=mat.size();//
        int n=mat[0].size();
        vector<vector<int>> states(m,vector<int>(n,-1));
        int dx[4]={1,0,0,-1};
        int dy[4]={0,1,-1,0};

        for(int i=0;i<m;i++)
        {
            for(int j=0;j<n;j++)
            {
                if(mat[i][j]==0)//从终点开始找
                {
                    qu.push({i,j});
                    states[i][j]=0;
                }
            }
        }

        while(!qu.empty())
        {
            int x1=qu.front().first;
            int y1=qu.front().second;

            qu.pop();
            for(int i=0;i<4;i++)
            {
                int x2=x1+dx[i];
                int y2=y1+dy[i];

                if(x2<0||x2>=m||y2<0||y2>=n)
                continue;
                if(mat[x2][y2]==0||states[x2][y2]!=-1)//同为0或者走过了先走的肯定更小
                continue;
                
                states[x2][y2]=states[x1][y1]+1;
                qu.push({x2,y2});

            }

        }
        return states;
    }
    private:
    queue<pair<int,int>> qu;
    
};
 飞地的数量

1020. 飞地的数量 - 力扣(LeetCode)

思路

解法1:

(非多源bfs)遍历地图,将遇到的并且没有走过的1进行bfs,如果没有陆地在边缘返回值为连通的陆地数量,在边缘就返回0,将返回值累加起来即可。

解法2:
(多源bfs)先考虑什么可以当源头,所有的1明显不可以,0进去也得不出来结果。我们发现题目的关键点在于边缘的陆地就不是飞地,所以我们将所有在边缘的陆地入队,然后查找与这些陆地相连的其他陆地将其标记,最后遍历地图查找没被标记的陆地即可。

ac代码

class Solution {
public:

    int bfs(vector<vector<int>>& grid,int x,int y)
    {
        int m=grid.size();
        int n=grid[0].size();
       
        int dx[4]={1,0,0,-1};
        int dy[4]={0,1,-1,0};

        queue<pair<int,int>> qu;
        qu.push({x,y});
        states[x][y]=true;
        int cnt=1;//记录有多少个陆地
        bool f=true;//如果有边缘就变false
        while(!qu.empty())
        {
            int x1=qu.front().first;
            int y1=qu.front().second;

            qu.pop();
            for(int i=0;i<4;i++)
            {
                int x2=x1+dx[i];
                int y2=y1+dy[i];

                if(x2<0||x2>=m||y2<0||y2>=n)
                {
                    f=false;//有靠边缘的
                    continue;
                }
                if(states[x2][y2]!=false||grid[x2][y2]==0)
                continue;

                states[x2][y2]=true;//走过
                qu.push({x2,y2});
                cnt++;

            }
        }
        if(f)
        return cnt;
        else
        return 0;
    }

    int numEnclaves(vector<vector<int>>& grid) {
        int m=grid.size();
        int n=grid[0].size();
        memset(states,false,sizeof states);
        int sum=0;
        for(int i=0;i<m;i++)
        {
            for(int j=0;j<n;j++)
            {
                if(grid[i][j]==1&&!states[i][j])
                {
                    sum+=bfs(grid,i,j);
                }
            }
        }
        return sum;
    }

    private:
    bool states[510][510];
};
地图分析

1162. 地图分析 - 力扣(LeetCode)

 思路

统计海洋单位与离其最近的陆地的距离,直接将所有陆地视为源头四散记录自己离海洋的距离,即可。每到一个海洋就比较到这个海洋的距离与之前记录的到海洋的距离那个更远。

ac代码

class Solution {
public:

    int maxDistance(vector<vector<int>>& grid) {
        int n=grid.size();
        int dx[4]={1,0,0,-1};
        int dy[4]={0,-1,1,0};
        vector<vector<int>> states(n,vector<int>(n,-1));
        for(int i=0;i<n;i++)
        {
            for(int j=0;j<n;j++)
            {
                if(grid[i][j]==1)
                {
                    qu.push({i,j});
                    states[i][j]=0;
                }
            }
        }

        while(!qu.empty())
        {
            int x1=qu.front().first;
            int y1=qu.front().second;

            qu.pop();

            for(int i=0;i<4;i++)
            {
                int x2=x1+dx[i];
                int y2=y1+dy[i];

                if(x2<0||x2>=n||y2<0||y2>=n)
                continue;
                if(grid[x2][y2]==1||states[x2][y2]!=-1)//不是陆地,不能走过
                continue;
                
                states[x2][y2]=states[x1][y1]+1;
                cntmax=max(cntmax,states[x2][y2]);
                qu.push({x2,y2});
                
            }
        }
        return cntmax;

    }

    private:
    int cntmax=-1;
    queue<pair<int,int>> qu;//存陆地
    
};
地图中的最高点

1765. 地图中的最高点 - 力扣(LeetCode)

思路 

题目很直观,将所有的水域入队,然后四处扩散,增加即可。

ac代码

class Solution {
public:
    vector<vector<int>> highestPeak(vector<vector<int>>& isWater) {
        
        int m=isWater.size();//
        int n=isWater[0].size();
        vector<vector<int>> states(m,vector<int>(n,-1));
        int dx[4]={1,0,0,-1};
        int dy[4]={0,1,-1,0};

        for(int i=0;i<m;i++)
        {
            for(int j=0;j<n;j++)
            {
                if(isWater[i][j]==1)//从水开始
                {
                    qu.push({i,j});
                    states[i][j]=0;//水域为0
                }
            }
        }

        while(!qu.empty())
        {
            int x1=qu.front().first;
            int y1=qu.front().second;

            qu.pop();
            for(int i=0;i<4;i++)
            {
                int x2=x1+dx[i];
                int y2=y1+dy[i];

                if(x2<0||x2>=m||y2<0||y2>=n)
                continue;
                if(isWater[x2][y2]==1||states[x2][y2]!=-1)//都为水域或者走过了先走的肯定更小
                continue;
                
                states[x2][y2]=states[x1][y1]+1;
                qu.push({x2,y2});

            }

        }
        return states;
    }
    private:
    queue<pair<int,int>> qu;        
};

2. 拓扑排序问题

将问题变成一个有向无环图,解决拓扑排序要找到做事情的先后顺序,结果可能不是唯一的。

如下图

有几个通向自己的节点入度就为几

我们找到图中入度为0的点a,必须将a消除后使b和c的入度也从1变为0才可以消除b和c,必须把b和c都消除后才能将d的入度变为0,再消除b。

实现拓扑排序的具体步骤:

借助队列:queue

1. 初始化:把所有入度为0的点加入到队列中

2. 当队列不为空时:

(1)拿出队头元素加到最终结果上

(2)删除与该元素相连的边

(3)判断删除边相连的点是否入度变为了0,为0就入队

可以创建邻接矩阵来构造图

也可以创建邻接表来构造:vector<vector<int>> 或者 unordered_map<int,vector<int>>

课程表

207. 课程表 - 力扣(LeetCode)

思路

创建一个哈希表来记录每个课程之间的关系将完成x课程的前置课程y1,y2等都存储起来,创建一个数组记录每个课程的入度

ac代码

class Solution {
public:
    bool canFinish(int numCourses, vector<vector<int>>& prerequisites) {
        in.resize(numCourses,0);
        int m=prerequisites.size();


        for(int i=0;i<m;i++)
        {
            //数组里只有两个数
            map[prerequisites[i][1]].push_back(prerequisites[i][0]);//b[i]通向a[i]
            in[prerequisites[i][0]]++;//a[i]多一个前置条件
        }
        for(int i=0;i<in.size();i++)
        {
            if(in[i]==0)//没前置条件
            qu.push(i);//将这门课程入队
        }
        int cnt=0;
        while(!qu.empty())
        {
            int tmp=qu.front();
            qu.pop();
            cnt++;//

            for(int i=0;i<map[tmp].size();i++)//删除这个前置,其链接的点少一个前置
            {
                in[map[tmp][i]]--;
                if(in[map[tmp][i]]==0)//前置都完成了
                {
                    qu.push(map[tmp][i]);
                }
            }
        }
        if(cnt==numCourses)
        return true;
        else
        return false;
    }

    private:
    unordered_map<int,vector<int>> map;//邻接表
    vector<int> in;//记录每个课程的入度
    queue<int> qu; 
};
课程表 II

LCR 113. 课程表 II - 力扣(LeetCode)

思路

与上题没什么不同,就是多一个数组来记录可以学习的课程

ac代码

class Solution {
public:
    vector<int> findOrder(int numCourses, vector<vector<int>>& prerequisites) {
        in.resize(numCourses,0);
        int m=prerequisites.size();

        for(int i=0;i<m;i++)
        {
            map[prerequisites[i][1]].push_back(prerequisites[i][0]);//将b[i]后置保存起来
            in[prerequisites[i][0]]++;//这个课程前置加1
        }
        for(int i=0;i<in.size();i++)
        {
            if(in[i]==0)
            qu.push(i);
        }
        vector<int> nums;
        while(!qu.empty())
        {
            int tmp=qu.front();
            qu.pop();
            nums.push_back(tmp);
            for(int i=0;i<map[tmp].size();i++)
            {
                in[map[tmp][i]]--;
                if(in[map[tmp][i]]==0)
                qu.push(map[tmp][i]);
            }
        }
        if(nums.size()==numCourses)
        return nums;
        else
        nums.resize(0,0);
        return nums;
    }
    unordered_map<int,vector<int>> map;//邻接表
    vector<int> in;//记录自身有多少前置
    queue<int> qu;
};
火星词典

LCR 114. 火星词典 - 力扣(LeetCode)

 思路

如何建图?
将第一个字符串与之后所有字符串对比(双指针一个字符一个字符对比)然后再将第二个字符串与第二个字符串之后所有字符串对比依次类推...,有不一样的就检测哈希表中x字符有没有已经存入相同的前置字符,没有再存进去,然后break再往后也找不出来了;要注意当数组走完并且第二个数组比第一个数组小时是错误的,要特判一下

记录入度

用一个哈希表进行记录

for(auto& [a,b]:in)//in中char会存到a里入度的值会存到b里
//用来遍历in数组

解决问题与上方差不多

ac代码

class Solution {
public:
    string alienOrder(vector<string>& words) {
        //所有前置走完走接下来的
        int m = words.size();

        for (auto& tmp : words)//初始化入度
        {
            for (auto t : tmp)
            {
                in[t] = 0;
            }
        }

        for (int i = 0; i < m - 1; i++)
        {
            for (int j = i + 1; j < m; j++)
            {
                int n=min(words[i].size(),words[j].size());
                string s1=words[i];
                string s2=words[j];
                int t=0;
               for( t=0;t<n;t++)
               {
                    if(s1[t]!=s2[t])//不相等再存
                    {
                        if(!map[s1[t]].count(s2[t]))//看s1[t]其中有没有存s2[t]
                        {
                            map[s1[t]].insert(s2[t]);
                            in[s2[t]]++;
                        }
                    break;//发现不同就不用往后找了
                    }
               }
               if(t==s2.size()&&t<s1.size())
               return "";
            }
        }
        int cnt=0;
        for(auto& [a,b]:in)//in中char会存到a里入度的值会存到b里
        {
            if(b==0) qu.push(a);
            cnt++;
        }
        string s = "";
        while (!qu.empty())
        {
            char tmp = qu.front();
            qu.pop();
            s += tmp;

            for (auto ch : map[tmp])//遍历哈希表
            {
                in[ch]--;
                if (in[ch] == 0)
                    qu.push(ch);
            }
        }
        if(s.size()==cnt)
        return s;
        else
        return "";

    }



    unordered_map<char, unordered_set<char>> map;//
    unordered_map<char, int> in;//记录这个单词还有多少单词
    queue<char> qu;
};

这篇就到这里了(๑′ᴗ‵๑)I Lᵒᵛᵉᵧₒᵤ❤

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Java基础--类和对象

目录 什么是类&#xff1f; 什么是对象 为什么java会设计对象 Java对象该怎么用 程序执行流程 类的加载顺序 什么是类&#xff1f; 类是构建对象的模板&#xff0c;一个类可以创建多个对象&#xff0c;每个对象的数据的最初来源来自对象 public class Student{public in…

学习ASP.NET Core的身份认证(基于JwtBearer的身份认证6)

重新创建WebApi项目&#xff0c;安装Microsoft.AspNetCore.Authentication.JwtBearer包&#xff0c;将之前JwtBearer测试项目中的初始化函数&#xff0c;jwt配置类、token生成类全部挪到项目中。   重新编写login函数&#xff0c;之前测试Cookie和Session认证时用的函数适合m…

Electron实践继续

文章目录 前言一、知识储备前提二、开发工具集&#xff08;一&#xff09;代码编辑器之选&#xff08;二&#xff09;命令行工具运用&#xff08;三&#xff09;Git 与 GitHub 协作利器&#xff08;四&#xff09;Node.js 与 npm 核心环境 你的第一个Electron应用程序 前言 上…

《自动驾驶与机器人中的SLAM技术》ch8:基于预积分和图优化的紧耦合 LIO 系统

和组合导航一样&#xff0c;也可以通过预积分 IMU 因子加上雷达残差来实现基于预积分和图优化的紧耦合 LIO 系统。一些现代的 Lidar SLAM 系统也采用了这种方式。相比滤波器方法来说&#xff0c;预积分因子可以更方便地整合到现有的优化框架中&#xff0c;从开发到实现都更为便…

【CSS】---- CSS 实现超过固定高度后出现展开折叠按钮

1. 实现效果 2. 实现方法 使用 JS 获取盒子的高度&#xff0c;来添加对应的按钮和样式&#xff1b;使用 CSS 的浮动效果&#xff0c;参考CSS 实现超过固定高度后出现展开折叠按钮&#xff1b;使用容器查询 – container 语法&#xff1b;使用 clamp 函数进行样式判断。 3. 优…

【C语言】_字符串拷贝函数strcpy

目录 1. 函数声明及功能 2. 使用示例 3. 注意事项 4. 模拟实现 4.1 第一版&#xff1a;基本功能判空const修饰 4.2 第二版&#xff1a;优化对于\0的单独拷贝 4.3 第三版&#xff1a;仿strcpy的char*返回值 1. 函数声明及功能 char * strcpy ( char * destination, cons…

大文件上传服务-后端V1V2

文章目录 大文件上传概述:minio分布式文件存储使用的一些技术校验MD5的逻辑 uploadV1 版本 1uploadv2 版本 2 大文件上传概述: 之前项目做了一个文件上传的功能,最近看到有面试会具体的问这个上传功能的细节&#xff0c;把之前做的项目拿过来总结一下&#xff0c;自己写的一个…

【k8s面试题2025】1、练气期

主要通过呼吸吐纳等方法&#xff0c;将外界的天地灵气吸入体内&#xff0c;初步改造身体&#xff0c;使身体素质远超常人。 文章目录 docker 和虚拟机的不同Kubernetes 和 docker 的关系Kube-proxy IPVS 和 iptables 的异同蓝绿发布Kubernetes中常见的数据持久化方式关于 Docke…

快速入门:如何注册并使用GPT

文章目录 ProtonMail邮箱步骤 1&#xff1a;访问Proton官网步骤 2&#xff1a;创建ProtonMail账户步骤 3&#xff1a;选择注册免费账户步骤 4&#xff1a;填写邮箱地址和手机号&#xff08;可选&#xff09;步骤 5&#xff1a;邮箱验证&#xff08;必须进行验证&#xff09;步骤…

浅谈云计算22 | Kubernetes容器编排引擎

Kubernetes容器编排引擎 一、Kubernetes管理对象1.1 Kubernetes组件和架构1.2 主要管理对象类型 二、Kubernetes 服务2.1 服务的作用与原理2.2 服务类型 三、Kubernetes网络管理3.1 网络模型与目标3.2 网络组件3.2.1 kube-proxy3.2.2 网络插件 3.3 网络通信流程 四、Kubernetes…

Vulnhub DC-8靶机攻击实战(一)

导语   Vulnhub DC-8靶机教程来了,好久没有更新打靶的教程了,这次我们在来更新一期关于Vulnhub DC-8的打靶训练,如下所示。 安装并且启动靶机 安装并且启动靶机,如下所示。 开始信息采集 进入到Kali中,通过如下的命令来查找到靶机的IP地址。 arp-scan -l根据上面的结…

自学SpringBoot笔记

概念 什么是SpringBoot&#xff1f; Spring Boot 是基于 Spring Framework 的一款开源框架&#xff0c;主要用于简化 Spring 应用程序的开发。它通过提供一系列的 开箱即用的功能 和 自动配置&#xff0c;让开发者可以快速构建生产级别的独立应用程序&#xff0c;而无需手动配…

通过学习更多样化的生成数据进行更广泛的数据分发来改进实例分割

大家读完觉得有帮助记得关注和点赞&#xff01;&#xff01;&#xff01; 本次使用的英文整理的一些记录&#xff0c;练习一下为后续SCI发表论文打好基础 Improving Instance Segmentation by Learning Wider Data Distribution with More Diverse Generative Data Abstract In…

USB3020任意波形发生器4路16位同步模拟量输出卡1MS/s频率 阿尔泰科技

信息社会的发展&#xff0c;在很大程度上取决于信息与信号处理技术的先进性。数字信号处理技术的出现改变了信息 与信号处理技术的整个面貌&#xff0c;而数据采集作为数字信号处理的必不可少的前期工作在整个数字系统中起到关键 性、乃至决定性的作用&#xff0c;其应用已经深…

提示词的艺术----AI Prompt撰写指南(个人用)

提示词的艺术 写在前面 制定提示词就像是和朋友聊天一样&#xff0c;要求我们能够清楚地表达问题。通过这个过程&#xff0c;一方面要不断练习提高自己地表达能力&#xff0c;另一方面还要锻炼自己使用更准确精炼的语言提出问题的能力。 什么样的提示词有用&#xff1f; 有…

自定义提示确认弹窗-vue

最初可运行代码 弹窗组件代码&#xff1a; &#xff08;后来发现以下代码可运行&#xff0c;但打包 typescript 类型检查出错&#xff0c;可打包的代码在文末&#xff09; <template><div v-if"isVisible" class"dialog"><div class&quo…

【JavaEE】Spring(1)

一、什么是Spring和SpringBoot Spring是Java应用程序的开发框架&#xff0c;其目的就是为了简化Java开发&#xff1b;SpringBoot是在spring框架的基础上构建的一个快速开发框架&#xff0c;其作用是进一步简化Spring程序开发 二、SpringBoot项目 2.1 创建项目 1. 设置jdk版本…

【Rust自学】13.4. 闭包 Pt.4:使用闭包捕获环境

13.4.0. 写在正文之前 Rust语言在设计过程中收到了很多语言的启发&#xff0c;而函数式编程对Rust产生了非常显著的影响。函数式编程通常包括通过将函数作为值传递给参数、从其他函数返回它们、将它们分配给变量以供以后执行等等。 在本章中&#xff0c;我们会讨论 Rust 的一…

浅谈操作系统与初识Linux

一、Linux操作系统的出现 1.1操作系统的出现以及相关的四个要素 1.2最早出现的操作系统及其创始人 起初&#xff0c;IBM为了让计算机可以以更低技术成本进行使用&#xff0c;以此来售卖计算机&#xff1b; 为计算机搭载上了Unix操作系统&#xff0c;Unix由肯汤普森用汇编语…

K8S开启/关闭审计日志

K8S默认禁用审计 开启/关闭 k8s 审计日志 默认 Kubernetes 集群不会输出审计日志信息。通过以下配置&#xff0c;可以开启 Kubernetes 的审计日志功能。 准备审计日志的 Policy 文件配置 API 服务器&#xff0c;开启审计日志重启并验证 准备审计日志 Policy 文件 apiVersio…