【优选算法专栏】专题十六:BFS解决最短路问题(二)

本专栏内容为:算法学习专栏,分为优选算法专栏,贪心算法专栏,动态规划专栏以及递归,搜索与回溯算法专栏四部分。 通过本专栏的深入学习,你可以了解并掌握算法。

💓博主csdn个人主页:小小unicorn
⏩专栏分类:算法从入门到精通
🚚代码仓库:小小unicorn的代码仓库🚚
🌹🌹🌹关注我带你学习编程知识

专题十六

  • 最小基因变化
    • 算法原理
  • 单词接龙
    • 算法原理:
  • 为高尔夫比赛砍树
    • 算法原理:

最小基因变化

题目来源:Leetcode433.最小基因变化

基因序列可以表示为一条由 8 个字符组成的字符串,其中每个字符都是 ‘A’、‘C’、‘G’ 和 ‘T’ 之一。

假设我们需要调查从基因序列 start 变为 end 所发生的基因变化。一次基因变化就意味着这个基因序列中的一个字符发生了变化。

例如,“AACCGGTT” --> “AACCGGTA” 就是一次基因变化。
另有一个基因库 bank 记录了所有有效的基因变化,只有基因库中的基因才是有效的基因序列。(变化后的基因必须位于基因库 bank 中)

给你两个基因序列 start 和 end ,以及一个基因库 bank ,请你找出并返回能够使 start 变化为 end 所需的最少变化次数。如果无法完成此基因变化,返回 -1 。

注意:起始基因序列 start 默认是有效的,但是它并不一定会出现在基因库中。
在这里插入图片描述

算法原理

本题目首先可以进行转化:

->转化为图论问题中的边权为1的最短路问题。
在这里插入图片描述

根据基因变换只能变换一个,我们把每一个基因抽象成一个点,那么这个问题就可以转化为:已知起点位置和终点位置,从起点位置到终止位置的最短路径,由于每次只能变换一个字母,所以边权都为1。

分析到这,解题思路就是用BFS。

但是在写代码前,还需注意一下细节:
在这里插入图片描述

  1. 为防止有的遍历重复,我们需要用一个数组来进行标记,但是普通数组显然不能满足需求,因为存的是字符串,所以我们应用哈希表进行储存。

  2. 如何枚举出所有的情况呢?
    答案很简单,我们暴力枚举一下字符串即可。

  3. 枚举出的所以情况,都需要加入到队列里么?
    答案是不需要,因为本身存在一个基因库,在枚举字符串中,在基因库中我们才进入队列。

  4. 如何快速判断在基因库中存在?
    普通做法就是直接遍历整个基因库,但是这样还是显得太麻烦,我们可以先预处理一下,直接将基因库丢在哈希中,在哈希表中进行查找,方便快速加快效率。

代码实现:

class Solution 
{
public:int minMutation(string startGene, string endGene, vector<string>& bank) {//用来标记是否是已经搜索过的状态unordered_set<string> vis;unordered_set<string> hash(bank.begin(),bank.end());//存储基因库中里面的字符串string change="ACGT";//处理边界情况if(startGene==endGene) return 0;int ret=0;queue<string> q;q.push(startGene);vis.insert(startGene);while(q.size()){ret++;int sz=q.size();while(sz--){string t=q.front();q.pop();for(int i=0;i<8;i++){//细节问题string tmp=t;for(int j=0;j<4;j++){tmp[i]=change[j];if(hash.count(tmp)&&!vis.count(tmp)){if(tmp==endGene)return ret;q.push(tmp);vis.insert(tmp);}}}}}return -1;}
};

单词接龙

题目来源:108.单词接龙

在字典(单词列表) wordList 中,从单词 beginWord 和 endWord 的 转换序列 是一个按下述规格形成的序列:

序列中第一个单词是 beginWord 。
序列中最后一个单词是 endWord 。
每次转换只能改变一个字母。
转换过程中的中间单词必须是字典 wordList 中的单词。
给定两个长度相同但内容不同的单词 beginWord 和 endWord 和一个字典 wordList ,找到从 beginWord 到 endWord 的 最短转换序列 中的 单词数目 。如果不存在这样的转换序列,返回 0。

在这里插入图片描述

算法原理:

分析一遍,其实本题跟最小基因库基本是一样的,只是本题的问法不一样:
本题问的是个数,所以我们要再返回层数的基础上加1.
在这里插入图片描述
代码实现:

class Solution 
{
public:int ladderLength(string beginWord, string endWord, vector<string>& wordList) {//存储单词列表unordered_set<string> hash(wordList.begin(),wordList.end());unordered_set<string> vis;//标记已经搜索过的单词//判断边界情况if(!hash.count(endWord)) return 0;queue<string> q;q.push(beginWord);vis.insert(beginWord);//因为返回的是个数,所以初始化可以初始化为1int ret=1;while(q.size()){ret++;int sz=q.size();while(sz--){string t=q.front();q.pop();for(int i=0;i<t.size();i++){string tmp=t;for(char ch='a';ch<='z';ch++){tmp[i]=ch;if(!vis.count(tmp)&&hash.count(tmp)){if(tmp==endWord) return ret;q.push(tmp);vis.insert(tmp);}}}}}return 0;}
};

为高尔夫比赛砍树

题目来源:Leetcode675.为高尔夫比赛砍树

你被请来给一个要举办高尔夫比赛的树林砍树。树林由一个 m x n 的矩阵表示, 在这个矩阵中:

0 表示障碍,无法触碰
1 表示地面,可以行走
比 1 大的数 表示有树的单元格,可以行走,数值表示树的高度
每一步,你都可以向上、下、左、右四个方向之一移动一个单位,如果你站的地方有一棵树,那么你可以决定是否要砍倒它。

你需要按照树的高度从低向高砍掉所有的树,每砍过一颗树,该单元格的值变为 1(即变为地面)。

你将从 (0, 0) 点开始工作,返回你砍完所有树需要走的最小步数。 如果你无法砍完所有的树,返回 -1 。

可以保证的是,没有两棵树的高度是相同的,并且你至少需要砍倒一棵树。
在这里插入图片描述

算法原理:

本题首先也可以转化:
可以看这个例子:
在这里插入图片描述

根据砍树由低到高,我们可以把砍树的顺序弄出来:
在这里插入图片描述
然后我们让每一段的路径都最小,那么总和肯定也就最小。
那么这个问题就可以转化为我们之前做过的迷宫问题。
而本题是若干个迷宫问题。

还记的什么是迷宫问题吗?
迷宫问题从起始点位置到边界情况下的最短路径,此题是前一个位置到后一个位置的最短路径。

问题解决到这,那么我们又怎么知道砍树顺序呢?
这时就需要用到我们的容器,我们用一个二维数组存储,可以是哈希可以是vector只要能将里面进行排序,那么我们就可以拿来用。

代码解决:

class Solution 
{int dx[4]={0,0,1,-1};int dy[4]={1,-1,0,0};public:int cutOffTree(vector<vector<int>>& f) {//int m = f.size(), n= f[0].size();//1.准备工作,找出砍树的顺序vector<pair<int,int>> tree;for(int i=0;i<m;i++){for(int j=0;j<n;j++){if(f[i][j]>1)tree.push_back({i,j});}}//排序sort(tree.begin(),tree.end(),[&](const pair<int,int>&p1,const pair<int,int>&p2){return f[p1.first][p1.second]<f[p2.first][p2.second];});//2.按照顺序砍树int bx=0,by=0;int ret=0;for(auto& [a,b]:tree){int step=bfs(f,bx,by,a,b);if(step==-1)return -1;ret+=step;bx=a,by=b;}return ret;}//3.bfsbool vis[51][51];int bfs(vector<vector<int>>& f,int bx,int by,int ex,int ey){int m = f.size(), n= f[0].size();if(bx==ex&&by==ey) return 0;queue<pair<int,int>> q;//清空之前数据memset(vis,0,sizeof vis);q.push({bx,by});vis[bx][by]=true;int step=0;while(q.size()){step++;int sz=q.size();while(sz--){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&&f[x][y]&&!vis[x][y]){if(x==ex&&y==ey)return step;q.push({x,y});vis[x][y]=true;}}}}return -1;}};

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

7.1.4 Selenium 爬取京东商品信息实战

目录 1、实战内容 2、思路 3、分析 url 4、开始操作 1、得到 Cookies 2、访问页面&#xff0c;得到 response 3、解析页面 4、存入 MySQL 5、1-3步总代码 1、实战内容 爬取京东笔记本电脑商品的信息(如&#xff1a;价格、商品名、评论数量)&#xff0c;存入 MySQL 中…

使用 AI 生成正则表达式,告别正则烦恼

如果你有处理正则表达式的需求&#xff0c;那么这个网站&#xff08;autoregex.xyz&#xff09;一定要收藏好。 可以根据文字描述生成正则表达式。 默认是从文字到正则&#xff0c;不用选择。 输入框中输入描述&#xff0c;点击 ”GO“ 按钮。 等待一会儿&#xff0c;即可生…

数据结构与算法:哈希表

目录 1.哈希表和哈希 1.1.知识引入 1.2.为什么需要哈希表呢&#xff1f; 2.简易的哈希表 2.1.哈希表的基础结构 2.2.如何实现基础的哈希表 2.2.1.增 2.2.2.删 2.2.3.查 2.3.泛型编程下的哈希表 3.简易的哈希桶 1.哈希表和哈希 1.1.知识引入 哈希表&#xff08;Hash …

11-新热文章-实时计算

热点文章-实时计算 1 今日内容 1.1 定时计算与实时计算 1.2 今日内容 kafkaStream 什么是流式计算 kafkaStream概述 kafkaStream入门案例 Springboot集成kafkaStream 实时计算 用户行为发送消息 kafkaStream聚合处理消息 更新文章行为数量 替换热点文章数据 2 实时…

网络通信三要素:IP、端口和协议

IP&#xff1a;设备在网络中的地址&#xff0c;是唯一的标识 IP&#xff1a;全程”互联网协议地址“&#xff0c;是分配给上网设备的唯一标志 IP地址有两种形式&#xff1a; IPv4&#xff1a;32位 IPv6&#xff1a;共128位。分成8段表示&#xff0c;每取四位编码成一个16进制…

Linux 5.10 Pstore 功能测试

目录 简介环境配置内核配置参考备注 简介 Pstore(Persistent store support)是用于系统发生oops或panic时&#xff0c;自动保存内核log buffer中的日志。随着功能不断完善&#xff0c;Duo S使用Linux 5.10已经支持保存console日志、ftrace消息和用户空间日志的收集&#xff0c…

Python学习从0开始——007输入与输出

Python学习从0开始——007输入与输出 一、简单输出二、复杂输出2.1引用变量2.2format()函数2.3手动格式化 三、读写文件3.1open()3.2操作文件对象3.3使用 json 保存结构化数据 一、简单输出 str() 函数返回供人阅读的值&#xff0c;repr() 则生成适于解释器读取的值&#xff0…

德兰梅尔:耐高温热销的膜元件亮相2024上海国际生物发酵展

德兰梅尔&#xff1a;耐高温热销的膜元件盛装亮相2024上海国际生物发酵展&#xff0c;8月7-9号上海新国际博览中心与您不见不散&#xff01; 据了解&#xff0c;从成立至今&#xff0c;德兰梅尔一直专注膜技术、膜产品的开发生产。在中国市场上&#xff0c;德兰梅尔刚步入中国…

峥嵘九载,逐云而上:青果乔迁新址,乘风破浪再起新篇

4月1日&#xff0c;近百名员工和诸多合作伙伴齐聚&#xff0c;共同见证了青果九周年庆典暨乔迁仪式这一里程碑式的时刻。 新起点&#xff0c;新征程&#xff0c;再启航&#xff01; 以新为序&#xff0c;共赴新征程 在典礼上&#xff0c;青果创始人和高管分别发表了致辞&#…

Python技能树学习-函数

题目一&#xff1a;递归调用 函数的参数&#xff1a; def dump(index, default0, *args, **kw): print(打印函数参数) print(---) print(index:, index) print(default:, default) for i, arg in enumerate(args): print(farg[{i}]:, arg) for…

力姆泰克医疗电动推杆

力姆泰克医疗电动推杆&#xff0c;宛如医疗领域的精巧舞者&#xff0c;以其卓越的性能和稳定的表现&#xff0c;为众多医疗应用献上了精彩的演绎。它犹如一位敬业的演员&#xff0c;始终保持着高度的灵敏度和精确度&#xff0c;无论是微调还是大幅移动&#xff0c;都能准确无误…

wireshark抓包新手使用教程

Wireshark是非常流行的网络封包分析软件&#xff0c;可以截取各种网络数据包&#xff0c;并显示数据包详细信息。常用于开发测试过程各种问题定位。本文主要内容包括&#xff1a; 1、Wireshark软件下载和安装以及Wireshark主界面介绍。 2、WireShark简单抓包示例。通过该例子学…