【基础算法总结】队列 + 宽搜(BFS)

队列 + 宽搜BFS

  • 1.N 叉树的层序遍历
  • 2.二叉树的锯齿形层序遍历
  • 3.二叉树最大宽度
  • 4.在每个树行中找最大值

在这里插入图片描述

点赞👍👍收藏🌟🌟关注💖💖
你的支持是对我最大的鼓励,我们一起努力吧!😃😃

1.N 叉树的层序遍历

题目链接: 429. N 叉树的层序遍历

题目分析:

在这里插入图片描述

层序遍历是一层一层的出,需要借助队列使用BFS来完成。比如说遍历到第三层的时候需要从第二层第一个节点的孩子先开始。因此就需要先记录这个节点的信息,遍历完3、4、5回过头再来遍历3的孩子。符合先进先出的思想。所以层序遍历要用到队列来实现。

在这里插入图片描述

算法原理:

解法:BFS(层次遍历/宽度优先搜索/宽度优先搜索)

层序遍历步骤,先搞一个队列,如果根节点不为空,现让根节点入队。接下来就是一个循环,当队列不空的时候,先把队头元素拿出来,然后把队头元素的孩子入队。循环下去直到队列为空为止。就把这棵树层序遍历结束了。

但是这道题要求的是一层一层出,因此我们需要一个变量在每次出队之前先统计一下当前队列中有多少个元素,然后每次出这一层的个数就行了。这个变量就是当前这一层元素的个数。

在这里插入图片描述

/*
// Definition for a Node.
class Node {
public:int val;vector<Node*> children;Node() {}Node(int _val) {val = _val;}Node(int _val, vector<Node*> _children) {val = _val;children = _children;}
};
*/class Solution {
public:vector<vector<int>> levelOrder(Node* root) {vector<vector<int>> ret; //记录最终结果if(root == nullptr) return ret;queue<Node*> qu;//层序遍历需要的队列qu.push(root);while(!qu.empty()){int num = qu.size();//先统计出本层元素的个数vector<int> tmp; //统计本层的节点while(num--){Node* front = qu.front();tmp.push_back(front->val);qu.pop();for(auto* cur : front->children)//下一层入队{qu.push(cur);}}ret.push_back(tmp);      }return ret;}
};

2.二叉树的锯齿形层序遍历

题目链接: 103. 二叉树的锯齿形层序遍历

题目分析:

在这里插入图片描述
锯齿形层序遍历 ,先从左往右,再从右往左进行下一层遍历,以此类推,层与层之间交替进行
在这里插入图片描述

算法原理:

解法:BFS

这道题和上面是一模一样的,只不过有一点小小的改变。发现偶数列是逆序放的。因此可以增加一个标记位,让偶数列的信息逆序即可。

/*** Definition for a binary tree node.* struct TreeNode {*     int val;*     TreeNode *left;*     TreeNode *right;*     TreeNode() : val(0), left(nullptr), right(nullptr) {}*     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}*     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}* };*/
class Solution {
public:vector<vector<int>> zigzagLevelOrder(TreeNode* root) {vector<vector<int>> ret;if(root == nullptr) return ret;queue<TreeNode*> qu;qu.push(root);int level = 1;while(!qu.empty()){int k = qu.size();vector<int> tmp;while(k--){TreeNode* t = qu.front();qu.pop();tmp.push_back(t->val);if(t->left) qu.push(t->left);if(t->right) qu.push(t->right);}if(level % 2 == 0) reverse(tmp.begin(),tmp.end());ret.push_back(tmp);++level;}return ret;}
};

3.二叉树最大宽度

题目链接:662. 二叉树最大宽度

题目分析:

在这里插入图片描述

每一层的 宽度 被定义为该层最左和最右的非空节点(即,两个端点)之间的长度。将这个二叉树视作与满二叉树结构相同,两端点间会出现一些延伸到这一层的 null 节点,这些 null 节点也计入长度。

最左和最右非空节点中间的nullptr也要加入计算到长度。
在这里插入图片描述

算法原理:

解法一:硬来

依旧是创建一个队列,这次入队不一样即使是一个节点的左孩子或者右孩子是null也要入队。然后在每一层出队之前都可以统计这一层有多宽。比如这棵树第三层没有问题宽度就是这一层的节点数,但是第四层就有问题了,实际宽度并不是这一层节点数。那如何统计这一层的实际宽度呢?方法很简单,我们在遍历这一层的时候可以再来一个empty指针,当这个指针遇到不是 null 的时候 宽度+1,如果是空的话就统计就循环把null拿到,如果碰到非空的时候就把 null 的个数 加入到 宽度,然后把null 的个数记为0。如果直到遍历到队列结尾还没有碰到非空 就不要把 null 结点数加入到 宽度。

在这里插入图片描述

但是这个方法会超时!
在这里插入图片描述

虽然节点有1-3000个,但是如果是系啊没这种左右非常平均有1500个节点。最后一层一共2^1499个节点,那太恐怖了。内存根本放不下。

在这里插入图片描述

接下来我们换一种策略。

树不仅可以用链式存储,也可以用数组存储起来,例如堆。

解法二:利用数组存储二叉树的方式,给结点编号

对一个结点i,如何找它的左右孩子,有两种方法如下:

在这里插入图片描述
有这个启发之后,那为何不给树每个结点都加上一个编号,孩子是null结点就不用在放入队列了,队列中只存储有孩子的结点和编号,然后这一层把第一个结点的编号和最后一个结点的编号拿出来,然后 最后一个结点的编号 减去 第一个结点的编号 + 1 不就是这一层宽度吗!

queue<pair<TreeNode*,int>>

还是这一层出队之前把第一个结点和最后一个结点拿到,取出里面的编号,做对应的计算。 如 7 - 4 + 1 = 4,14 - 8 + 1 = 7。比较每层宽度取最大的宽度。
在这里插入图片描述

上面是用队列来做的,甚至可以不用队列,就用两个数组就可以模拟队列。因此
vector<pair<TreeNode*,int>> 也是可以的。

这里还有一个细节问题:下标有可能会溢出。还是刚才的例子,左边来1500个节点,右边来1500个节点。最后一个节点编号是2^1500 - 1,不管用什么类型都存不下!别说这里int,甚至double都不行。

在这里插入图片描述

难道要用字符串来存,然后做高精度加减吗,其实也不用, 因为我们最后是做减法的,即使溢出,但是相减的结果也是正确的。

我们的数据存储其实是可以看作一个环,如char -128 ~ 127
虽然正数会溢出到负数, 但是 我们做减法求的是两个数之间的距离,当两个数相减就会把结果修正,因为求的是之间的距离,这段距离是正的并且不会溢出。

但是前提不能是超过一圈然后到的这个位置。此时减肯定是一个错误。但是这道题告诉题目数据保证答案将会在 32 位 带符号整数范围内。因此pair 内 int 换成 unsigned int

vector<pair<TreeNode*,unsigned int>>

在这里插入图片描述

/*** Definition for a binary tree node.* struct TreeNode {*     int val;*     TreeNode *left;*     TreeNode *right;*     TreeNode() : val(0), left(nullptr), right(nullptr) {}*     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}*     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}* };*/
class Solution {
public:int widthOfBinaryTree(TreeNode* root) {queue<pair<TreeNode*,unsigned int>> qu;unsigned int ret = 0;if(root == nullptr) return ret;qu.push(make_pair(root,1));while(!qu.empty()){   //先更新这一层的宽度//pari对象里面的成员会分别存到 first -> x1, second -> y1auto& [x1,y1] = qu.front();auto& [x2,y2] = qu.back();ret = max(ret, y2 - y1 + 1);//让下一层入队int k = qu.size();while(k--){auto [x,y] = qu.front();qu.pop();if(x->left) qu.push(make_pair(x->left,2 * y));if(x->right) qu.push(make_pair(x->right,2 * y +1));}}return ret;}
};

数组模拟队列

class Solution {
public:int widthOfBinaryTree(TreeNode* root) {vector<pair<TreeNode*, unsigned int>> q; // ⽤数组模拟队列q.push_back({root, 1});unsigned int ret = 0;while(q.size()){// 先更新这⼀层的宽度auto& [x1, y1] = q[0];auto& [x2, y2] = q.back();ret = max(ret, y2 - y1 + 1);// 让下⼀层进队//数组队头不好出,那就不出,再来一个数组记录孩子节点,最后拷贝回去vector<pair<TreeNode*, unsigned int>> tmp; // 让下⼀层进⼊这个队列for(auto& [x, y] : q){if(x->left) tmp.push_back({x->left, y * 2});if(x->right) tmp.push_back({x->right, y * 2 + 1});}q = tmp;}return ret;}
};

4.在每个树行中找最大值

题目链接:515. 在每个树行中找最大值

题目描述:

在这里插入图片描述

算法原理:

解法:利用层序遍历,统计出每一层的最大值

class Solution {
public:vector<int> largestValues(TreeNode* root) {vector<int> ret;if(root == nullptr) return ret;queue<TreeNode*> qu;qu.push(root);while(qu.size()){int k = qu.size();int tmp = INT_MIN;while(k--){TreeNode* t = qu.front();qu.pop();tmp = max(tmp, t->val);if(t->left) qu.push(t->left);if(t->right) qu.push(t->right);}ret.push_back(tmp);}return ret;}
};

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

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

相关文章

MySQL作业五

1. 创建表goods&#xff0c;orders 2. 向商品表中插入商品记录 3. 触发器操作 3.1 建立触发器&#xff0c;订单表中增加订单数量后&#xff0c;商品表商品数量同步减少对应的商品订单出数量,并测试 3.2 建立触发器&#xff0c;实现功能:客户取消订单&#xff0c;恢复商品表对应…

java数据结构之排序

前言&#xff1a; 排序在我们日常生活中随处可见&#xff0c;这里将介绍java数据结构里面常见的几种排序。 ps: swap函数的实现&#xff1a; public void swap(int[] arr, int i, int j) {int tmp arr[i];arr[i] arr[j];arr[j] tmp; } 1.直接插入排序 &#xff08;1&a…

类与对象(补充)

初始化列表 1. 之前我们实现构造函数时&#xff0c;初始化成员变量主要使用函数体内赋值&#xff0c;构造函数初始化还有一种方式&#xff0c;就是初始化列表&#xff0c;初始化列表的使用方式是以一个冒号开始&#xff0c;接着是一个以逗号分隔的数据成员列表&#xff0c;每个…

【OpenCV C++20 学习笔记】序列化——XML和YAML文件处理

序列化——XML和YAML文件处理 序列化和反序列化代码实现XML/YAML文件的打开和关闭写入或读取文本和数字写入或读取OpenCV数据写入或读取数组以及map读取和写入自定义数据类型 输出结果 序列化和反序列化 如果希望永久保存某些对象&#xff0c;而不是每次运行程序的时候重新创建…

经典文献阅读之--LIV-GaussMap(实时3D辐射场地图渲染的LiDAR惯性视觉融合算法)

Tip: 如果你在进行深度学习、自动驾驶、模型推理、微调或AI绘画出图等任务&#xff0c;并且需要GPU资源&#xff0c;可以考虑使用UCloud云计算旗下的Compshare的GPU算力云平台。他们提供高性价比的4090 GPU&#xff0c;按时收费每卡2.6元&#xff0c;月卡只需要1.7元每小时&…

如何优化网站以提升UX设计质量

什么叫 UX 设计&#xff1f;UX 设计&#xff0c;即用户体验设计&#xff0c;是指为提升用户体验而进行的产品设计。 UX 在设计中&#xff0c;设计师通过调查和研究用户来掌握用户的需求和喜好&#xff0c;并利用这些信息来设计产品。设计师还会测试产品&#xff0c;以确保它们能…

学习笔记之Java篇(0725)

p this 普通方法中&#xff0c;this总是指向调用该方法的对象。 构造方法中&#xff0c;this总是指向正要初始化的对象。 this&#xff08;&#xff09;调用必须重载的构造方法&#xff0c;避免相同地址初始化代码&#xff0c;但只能在构造方法中用&#xff0c;比企鹅必须位…

不让录制的屏幕如何绕开?轻松突破录屏限制:三招搞定App录屏难题

在数字时代&#xff0c;屏幕录制已成为分享知识和记录重要信息的必备技能。然而&#xff0c;有些应用程序出于版权保护或其他原因&#xff0c;限制了屏幕录制功能。这是否意味着我们束手无策呢&#xff1f;当然不是&#xff01;本文将为您揭秘三种简单易行的方法&#xff0c;让…

html+css前端作业 王者荣耀官网1个页面(带报告)

htmlcss前端作业 王者荣耀官网1个页面&#xff08;带报告&#xff09; 下载地址 https://download.csdn.net/download/qq_42431718/89575045 目录1 目录2 项目视频 王者荣耀首页1个页面&#xff08;无js&#xff09; 页面1

【QT】SARibbon编译安装开启frameless(QWindowkit)

1.cmake开启frameless 2.检查cmakecache 3.下载编译qwindowkit 拉取saribbon时请 git clone https://github.com/czyt1988/SARibbon.git --recursive使用--recursive可以拉取第三方库 手动下载&#xff1a;https://github.com/stdware/qwindowkit 4.cmake构建 和 visual stu…

2024-07-24 Linux C語言使用inotify进行文件变化检测

一、在Linux中&#xff0c;用C语言检测文件内容变化的方法有几种&#xff0c;最常用的包括以下几种&#xff1a; 轮询&#xff08;Polling&#xff09;&#xff1a;周期性地读取文件并检查内容是否变化。inotify&#xff1a;使用Linux内核提供的inotify接口&#xff0c;这是一…

Java项目中整合多个pdf合并为一个pdf

一、Java项目中整合多个pdf合并为一个pdf gitee笔记路径&#xff1a;https://gitee.com/happy_sad/drools一、依赖导入 <dependency><groupId>com.itextpdf</groupId><artifactId>itextpdf</artifactId><version>5.5.6</version> …

Centos8 yum 更换源以及安装内核头文件

文章目录 一、简介二、yum 更换源三、安装内核头文件 一、简介 CentOS 是一个开源项目&#xff0c;发布了两个不同的 Linux 发行版——CentOS Stream 和 CentOS Linux 。 CentOS Stream 是即将发布的红帽企业 Linux 产品的上游开发平台。 CentOS 项目将于 2024 年 6 月 30 日…

嵌入式C++、ROS 、OpenCV、SLAM 算法和路径规划算法:自主导航的移动机器人流程设计(代码示例)

在当今科技迅速发展的背景下&#xff0c;嵌入式自主移动机器人以其广泛的应用前景和技术挑战吸引了越来越多的研究者和开发者。本文将详细介绍一个嵌入式自主移动机器人项目&#xff0c;涵盖其硬件与软件系统设计、代码实现及项目总结&#xff0c;并提供相关参考文献。 项目概…

基于区块链技术的高校教育资源共享的研究

&#xff08;一&#xff09;项目背景 时代变迁下的高教管理革新需求 当前&#xff0c;我国高等教育体系深受行政化管理模式影响&#xff0c;其在指引办学方向、资源优化配置及院校稳定上功不可没。然而&#xff0c;随着社会主义市场经济体系的深化发展&#xff0c;该模式逐渐显…

电脑录屏直播怎么录?3款软件推荐,达人必备

电脑录屏直播成为了一种新型、有趣且高效的传播方式。想象一下&#xff0c;当您喜欢的游戏博主进行精彩有趣的游戏直播&#xff0c;而您却因为没时间将要错过这场精彩绝伦的直播。这时&#xff0c;一款好用的录屏软件是您的必需品&#xff0c;电脑录屏能让您不再错过屏幕上的精…

学习记录701@org.hibernate.MappingException: No Dialect mapping for JDBC

使用spring data jpa 时报错&#xff1a;javax.persistence.PersistenceException: org.hibernate.MappingException: No Dialect mapping for JDBC type: 0。 但是在数据库中sql是可以执行的。 我是用的是原生查询&#xff1a; Query query entityManager.createNativeQuer…

第一百八十一节 Java IO教程 - Java文件树

Java IO教程 - Java文件树 FileVisitor API可以递归地处理文件树中的所有文件和目录。 当我们要对文件树中的所有或某些文件或目录执行某些操作时&#xff0c;FileVisitor API非常有用。 SimpleFileVisitor类是FileVisitor接口的基本实现。 当访问文件/目录时&#xff0c;Si…

React Native在移动端落地实践

在移动互联网产品迅猛发展的今天&#xff0c;技术的不断创新使得企业越来越注重降低成本、提升效率。为了在有限的开发资源下迅速推出高质量、用户体验好的产品&#xff0c;以实现公司发展&#xff0c;业界催生了许多移动端跨平台解决方案。这些方案不仅简化了开发流程&#xf…

zookeeper开启SASL权限认证

目录 一、SASL介绍 二、使用 SASL 进行身份验证 2.1 服务器到服务器的身份验证 2.2 客户端到服务器身份验证 三、验证功能 一、SASL介绍 默认情况下&#xff0c;ZooKeeper 不使用任何形式的身份验证并允许匿名连接。但是&#xff0c;它支持 Java 身份验证与授权服务(JAAS)…