进阶了解C++(6)——二叉树OJ题

Leetcode.606.根据二叉树创建字符串:

606. 根据二叉树创建字符串 - 力扣(LeetCode)

难度不大,根据题目的描述,首先对二叉树进行一次前序遍历,即:

class Solution {
public:string tree2str(TreeNode* root) {if(root == nullptr){return "";}string ret = to_string(root->val);ret += tree2str(root->left);ret += tree2str(root->right);return ret;}
};

输出结果如下:

       从结果看到,元素输出的顺序正确,但是输出的格式不正确。对于题目要求的格式输出,不难发现,当一个结点的左子树的结点为空时,括号不输出。当一个结点的左子树的结点为空,但是右子树的结点不为空时,正常打印左,右结点的括号。只有当右结点存在时,才打印右结点的括号。

      对于或逻辑运算符'\left | \right |'的运算逻辑如下:a\left | \right |b\left | \right |c\left | \right |d假如a为真,则不在对下面的情况进行判断,假如a为假,则会挨个事件进行判断,如果a,b,c,d全为假才为假。在上面说到,对于一个结点的左子树根结点括号是否需要打印要分为两种情况:

      1.结点的左子树结点为空,但是结点的右子树结点不为空,此时正常对结点的左子树结点的括号进行打印。

      2.结点的左子树结点为空,同时右子树结点为空,此时不打印任何括号。

      因此,对于上面两种结点情况的判定,可以使用\left | \right |逻辑完成。首先判断root->left是否为空,如果不为空,则正常打印左结点的括号。 如果为空,则去判断root->right,如果不为空,则正常打印左结点的括号,如果为空,则不打印左结点的括号。

      对于右结点情况的判定,如果右结点为空,则不打印括号,如果不为空,则正常打印,因此,对应代码如下:
 

class Solution {
public:string tree2str(TreeNode* root) {if(root == nullptr){return "";}string ret = to_string(root->val);if(root->left || root->right){ret+='(';ret += tree2str(root->left);ret+=')';}if(root->right){ret+='(';ret += tree2str(root->right);ret+=')';}return ret;}
};

 运行结果如下:

Leetcode.102 二叉树的层序遍历:

102. 二叉树的层序遍历 - 力扣(LeetCode)

对于本题,可以利用队列来实现,具体思路如下:

    首先利用栈,将二叉树根结点所对应的元素压入队列,即:

      由于层序遍历的特点,因此在下一步,需要将3的左右子树的结点9,20进入队列。不过此时会面临一个问题,如何判断下一层全部进入队列?对于图中的树,第二层只有两个结点,姑且可以人为创建两次入队列的操作,但是对于下面层的结点,例如结点的数量为x个,不好判断入队列的次数。但是,对于这x个结点,因为其父结点的原因。是绝对可以在x/2次数内入完队列的。所以,对于入队列的次数,可以分成x/2次数次完成,即通过其父结点来完成。

    所以,额外定义一个变量levelsize来记录本层结点,即下一层结点的父结点的数量,例如对于上图中的队列,levelsize=1

   随后,利用循环,循环levlesize次,保证子结点都能入到队列中。由于题目要求的输出方式是类似于二维数组的方式,所以需要额外创建一个vector<int>类型的变量v用于向二维数组中不断插入层序遍历到的结点。

   对于上述步骤,仅仅用文字描写不够清晰,下面将利用图片进行演示:
   首先,层序遍历每一层的结点的结果在输出时是独立的,因此,在进入循环后,首先创建一个指针frontqueue中的结点的地址进行保存,并且将这个结点出队列。即:

然后,将这个结点的数值插入到v中,即:

随后,让front的左右结点依次入队列,即:

随后,令v插入到二维数组vv中,并且由于此时队列中存储了两个结点,所以改变levelsize,即:

至此,一次循环运行完毕。由于第一层的levelsize==1,因此上述步骤只会进行一次。为了便于理解,再给出第二次循环的相关图片解释。

此时队列中存储了9,20两个结点的地址。首先利用front结点保存队列中第一个结点的地址。即:

然后让队列头部元素出队列,并且将这个元素插入到v中,即:

随后,让front的左右结点依次进队列,即:

由于levelsize==2,因此,上述步骤还会进行一次,第二次具体如下:
利用front保存队列的头部元素,然后出队列的头部元素,再令v保存front->val即:

然后在令front左右结点的地址进入队列,即:

最后,令vv保存v,再改变levelsize,即:

对应代码如下:
 

class Solution {
public:vector<vector<int>> levelOrder(TreeNode* root) {vector<vector<int>> vv;//用于进出每一次的结点queue<TreeNode*> q;//记录每层结点数量int levelsize = 0;if(root){q.push(root);levelsize = 1;}while(!q.empty()){vector<int> v;while(levelsize--){TreeNode* front = q.front();q.pop();v.push_back(front->val);if(front->left)q.push(front->left);if(front->right)q.push(front->right);}levelsize = q.size();vv.push_back(v);}return vv;}
};

在上面的代码中,需要注意,由于v保存的是某一层的结点,与上层无关,因此,v定义在第一层while循环中,可以保证,每次循环执行完毕后,v都会进行一次自动的更新。

运行结果如下:

Leetcode.236 二叉树的最近公共祖先:

236. 二叉树的最近公共祖先 - 力扣(LeetCode)

本题可以说是本文中难度最大题目。对于题目中公共祖先的定义,可以分为下面的类型:
例如对于下面给出的二叉树中,结点7,0的最近公共祖先为根结点,即存储值为3的结点:

对于结点6,4的最近公共祖先为存储值为5的结点,即:

对于结点2,4的最近公共祖先是存储值为2的结点,即:

通过对于上面不同结点所对应的公共结点,可以得到寻找公共结点的规律:

1.如果给定的两个结点p,q分别在树的左子树和右子树,则根结点root就是最近公共结点。(对应第一个图所对应的情况)。

2.如果给定的两个结点p,q同时在树的左子树或者同时在树的右子树,(对应第二个图所对应的情况),此时,可以去判断p,q是否在根结点的子结点的左右,如果在,则根结点的子结点就是p,q的最近公共祖先,即:

3.如果p,q中的一个是另一个的子结点,例如假设pq的子结点,则q就是最近公共结点。

因此,在查找p,q的最近公共祖先之前,需要先确定p,q两个结点在树中的位置。二者的位置可以分为下面四种情况: p结点在左子树,p结点在右子树,q结点在左子树,q结点在右子树。为了方便表示,用pInleft,pInright,qInleft,qInright分别表示上面的四种情况。对于上面结点位置的查询,可以使用递归来实现,具体代码如下:

bool IsInTree(TreeNode* root, TreeNode* x){if(root == nullptr){return false;}return root == x || IsInTree(root->left,x) || IsInTree(root->right,x);}

同时,如果题目给定的树为空树,则不需要进行判断,如果给定的p,q其中之一就是根结点,则直接返回根结点即可。

在利用上面的代码确定了p,q结点在树中的位置后,需要分下面的情况进行判定:
入如果pInleft,qInright同时为真或者pInright,qInleft同时为真,则说明p,q结点分别分布在左子树或者右子树。因此对于这种情况,直接返回根结点即可。

如果pInleft,qInleft同时为真或者pInright,qInright同时为真,则说明p,q同时在左子树或者右子树。因此直接对其根结点左子树,或者右子树进行重复判断即可。对应代码如下:
 

class Solution {
public:bool IsInTree(TreeNode* root, TreeNode* x){if(root == nullptr){return false;}return root == x || IsInTree(root->left,x) || IsInTree(root->right,x);}TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {if(root == nullptr){return nullptr;}if(root == p || root == q){return root;}bool pInleft,pInright,qInleft,qInright;pInleft = IsInTree(root->left,p);pInright = ! pInleft;qInleft = IsInTree(root->left,q);qInright = ! qInleft;if((pInleft && qInright) || (pInright && qInleft)){return root;}else if((pInleft && qInleft) ){return lowestCommonAncestor(root->left,p,q);}else if((qInright && pInright)){return lowestCommonAncestor(root->right,p,q);}assert(false);return nullptr;}
};

运行结果如下:

但是这种方法时间复杂度太大。因此,可以采用下面的方法来优化时间复杂度:

例如对于上面的树,如果将查找两个结点的路径记录,则记录的结点如下:

则不难发现,所谓的最近公共祖先,就是二者路径上最近的相同结点。而重点,就是如何去记录这个路径。文章给出一种方法,具体如下:
首先将寻找路径的函数命名为GetPath,首先检测当前结点是否为空,如果为空,则直接返回false,如果不为空,首先创建一个栈,这里将这个栈命名为s,让结点入栈。在入栈后,检测当前结点是否是需要寻找的结点,如果是则返回true。如果不是,则分别递归结点的左右子树。如果在左右子树中没有找到,则出栈一次。为了方便理解,下面用图来演示这一过程:

首先,此时的结点3不为空,因此直接入栈,即:

由于3并不是需要找的结点,因此先去结点的左子树中进行寻找,即5。与上方相同的逻辑,首先让5入栈,即:

由于5也不是需要查找的结点,因此按照递归,再去左子树进行查找。

让结点6入栈,即:

由于6也不是需要找的结点,因此继续往左子树递归。由于6结点的左,右子树结点都为空,因此判定为左,右子树都找不到结点,所以将6弹出栈,即:

此时递归回到上一层,由于5结点的左子树找不到需要的结点,因此去其右子树进行查找,即将2入栈;


同时,由于2也不是需要找的结点,因此去其左子树进行查找,即将7入栈。由于此时的结点为需要查找的结点,因此返回true。此时栈中内容如下;

此时,便获取了查找结点的路径。

对于另一个需要查找的结点,由于原理相同,因此不再过多叙述。

在得到了两个结点的路径后,首相让长的路径逐渐pop,直到两个栈的size相同即可。随后挨个比较栈中元素。如果相同则返回即可。对应代码如下:
 

class Solution {
public:bool GetPath(TreeNode* root, TreeNode* x, stack<TreeNode*>& s){if(root == nullptr){return false;}s.push(root);if(root == x){return true;}if(GetPath(root->left,x,s)){return true;}if(GetPath(root->right,x,s)){return true;}s.pop();return false;}TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {stack<TreeNode*> sp;stack<TreeNode*> sq;GetPath(root,p,sp);GetPath(root,q,sq);while(sp.size() != sq.size()){if(sp.size() > sq.size()){sp.pop();}else{sq.pop();}}while(sp.top() != sq.top()){sp.pop();sq.pop();}return sp.top();}
};

运行结果如下:

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

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

相关文章

[flink 实时流基础系列]揭开flink的什么面纱基础一

Apache Flink 是一个框架和分布式处理引擎&#xff0c;用于在无边界和有边界数据流上进行有状态的计算。Flink 能在所有常见集群环境中运行&#xff0c;并能以内存速度和任意规模进行计算。 文章目录 0. 处理无界和有界数据无界流有界流 1. Flink程序和数据流图2. 为什么一定要…

Triton推理服务器部署YOLOv8实战

课程链接&#xff1a;Triton推理服务器部署YOLOv8实战_在线视频教程-CSDN程序员研修院 Triton Inference Server&#xff08;Triton 推理服务器&#xff09;是一个高性能、灵活、可扩展的推理服务器&#xff0c;支持多种机器学习框架&#xff08;PyTorch、ONNX等&#xff09;和…

计算机网络:物理层 - 信道极限容量

计算机网络&#xff1a;物理层 - 信道极限容量 实际信道中的数字信号奈式准则香农公式练习 实际信道中的数字信号 信号在传输过程中会受到各种因素的影响&#xff0c;如图所示&#xff1a; 这是一个数字信号&#xff0c;当它通过实际的信道后&#xff0c;波形会产生失真&#…

C# 高级文件操作与异步编程探索(初步)

文章目录 文本文件的读写探秘StreamReader 类深度剖析StreamWriter 类细节解读编码和中文乱码的解决方案 二进制文件的读写BinaryReader 类全面解析BinaryWriter 类深度探讨 异步编程与C#的未来方向同步与异步&#xff1a;本质解读Task 的神奇所在async/await 的魔法 在现代编程…

jvm(HotSpotVM)学习记录

1&#xff1a;当我们写的java文件经过编译器编译后生成class文件。class文件里面包含了各种java语言规范的特定语法。可以通过javap -v -private xxx.class 例如&#xff1a; javap -v -private Gas.class Classfile /D:/project/ike-springboot-carbonzero-v2/springboot-bms…

探索智慧农业精准除草,基于高精度YOLOv5全系列参数【n/s/m/l/x】模型开发构建农田作物场景下杂草作物分割检测识别分析系统

智慧农业是未来的一个新兴赛道&#xff0c;随着科技的普及与落地应用&#xff0c;会有更加广阔的发展空间&#xff0c;关于农田作物场景下的项目开发实践&#xff0c;在我们前面的博文中也有很堵相关的实践&#xff0c;单大都是偏向于目标检测方向的&#xff0c;感兴趣可以自行…

多焦点图像融合文献学习(一)

本文介绍的是一篇明为"A convolutional neural network-based conditional random field model for structured multi-focus image fusion robust to noise."的文献&#xff0c;主要包括文献的摘要、前言摘选、主要贡献、网络结构、实验结果及结论等方面。 文献名称摘…

CorelDRAW25.0.0.230最新2024版本Crack下载安装方法

CorelDRAW (CDR)是一款专业的平面设计软件。本软件是由加拿大Corel公司开发的一款功能强大的专业平面设计软件、矢量设计软件、矢量绘图软件。本矢量图形制作工具软件广泛应用于商标设计、标志制作、封面设计、CIS设计、产品包装设计、模型绘制、插画绘制、时装/服装设计、印刷…

Linux——信号的保存与处理

目录 前言 一、信号的常见概念 1.信号递达 2.信号未决 3.信号阻塞 二、Linux中的递达未决阻塞 三、信号集 四、信号集的处理 1.sig相关函数 2.sigprocmask()函数 3.sigpending()函数 五、信号的处理时机 六、信号处理函数 前言 在之前&#xff0c;我们学习了信号…

最近Sentinel-2下载网站好像有点问题

最近Sentinel-2下载网站好像有点点问题&#xff0c;基本属于打不开的状态&#xff01; 这也难怪&#xff0c;Sentinel提供了全世界最好的免费遥感资源。其受欢迎程度可以想象的到&#xff01;这么多人访问网站&#xff0c;网站压力可以说是巨大的。这可是全世界的并发访问&…

C/C++语言学习路线: 嵌入式开发、底层软件、操作系统方向(持续更新)

初级&#xff1a;用好手上的锤子 1 【感性】认识 C 系编程语言开发调试过程 1.1 视频教程点到为止 1.2 炫技视频看看就行 1.3 编程游戏不玩也罢 有些游戏的主题任务就是编程&#xff0c;游戏和实际应用环境有一定差异&#xff08;工具、操作流程&#xff09;&#xff0c;在…

Haproxy2.8.1+Lua5.1.4部署,haproxy.cfg配置文件详解和演示

目录 一.快速安装lua和haproxy 二.配置haproxy的配置文件 三.配置haproxy的全局日志 四.测试负载均衡、监控和日志效果 五.server常用可选项 1.check 2.weight 3.backup 4.disabled 5.redirect prefix和redir 6.maxconn 六.调度算法 1.静态 2.动态 一.快速安装lu…