进阶了解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();
        
    }
};

运行结果如下:

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

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

相关文章

TheMoon 恶意软件短时间感染 6,000 台华硕路由器以获取代理服务

文章目录 针对华硕路由器Faceless代理服务预防措施 一种名为"TheMoon"的新变种恶意软件僵尸网络已经被发现正在侵入全球88个国家数千台过时的小型办公室与家庭办公室(SOHO)路由器以及物联网设备。 "TheMoon"与“Faceless”代理服务有关联&#xff0c;该服务…

【算法题】三道题理解算法思想--滑动窗口篇

滑动窗口 本篇文章中会带大家从零基础到学会利用滑动窗口的思想解决算法题&#xff0c;我从力扣上筛选了三道题&#xff0c;难度由浅到深&#xff0c;会附上题目链接以及算法原理和解题代码&#xff0c;希望大家能坚持看完&#xff0c;绝对能有收获&#xff0c;大家有更好的思…

Flask学习(六):蓝图(Blueprint)

蓝图&#xff08;Blueprint&#xff09;&#xff1a;将各个业务进行区分&#xff0c;然后每一个业务单元可以独立维护&#xff0c;Blueprint可以单独具有自己的模板、静态文件或者其它的通用操作方法&#xff0c;它并不是必须要实现应用的视图和函数的。 Demo目录结构&#xf…

计算机专业学习单片机有什么意义吗?

玩单片机跟玩计算机区别还是很大的, 单片机有众多的种类,每一种又可能有很多个系列.可以说单片机就是为了专款专用而生的.这样来达到产品成本的降低,这就是现在身边的很多的电子产品价格一降再降的原因之一.在开始前我有一些资料&#xff0c;是我根据网友给的问题精心整理了一…

Python拆分PDF、Python合并PDF

WPS能拆分合并&#xff0c;但却是要输入编辑密码&#xff0c;我没有。故写了个脚本来做拆分&#xff0c;顺便附上合并的代码。 代码如下&#xff08;extract.py) #!/usr/bin/env python """PDF拆分脚本(需要Python3.10)Usage::$ python extract.py <pdf-fil…

腾讯云4核8g服务器多少钱?2024轻量和CVM收费价格表

2024年腾讯云4核8G服务器租用优惠价格&#xff1a;轻量应用服务器4核8G12M带宽646元15个月&#xff0c;CVM云服务器S5实例优惠价格1437.24元买一年送3个月&#xff0c;腾讯云4核8G服务器活动页面 txybk.com/go/txy 活动链接打开如下图&#xff1a; 腾讯云4核8G服务器优惠价格 轻…

uniapp 微信小程序 canvas 手写板获取书写内容区域并输出

uni.canvasGetImageData 返回一个数组&#xff0c;用来描述 canvas 区域隐含的像素数据&#xff0c;在自定义组件下&#xff0c;第二个参数传入自定义组件实例 this&#xff0c;以操作组件内 组件。 // 获取目标 canvas 的像素信息 pixelData let canvas uni.createSelector…

Linux 系统 CentOS7 上搭建 Hadoop HDFS集群详细步骤

集群搭建 整体思路:先在一个节点上安装、配置,然后再克隆出多个节点,修改 IP ,免密,主机名等 提前规划: 需要三个节点,主机名分别命名:node1、node2、node3 在下面对 node1 配置时,先假设 node2 和 node3 是存在的 **注意:**整个搭建过程,除了1和2 步,其他操作都使…

linux 内存介绍

大致共有四类&#xff1a;VSS、RSS、PSS、USS &#xff0c;通常情况下&#xff0c;VSS > RSS > PSS > USS 1.VSS(Virtual Set Size)虚拟耗用内存&#xff08;包含共享库占用的内存&#xff09; VSS表示一个进程可访问的全部内存地址空间的大小。这个大小包括了进程已…

Vue3使用vue-office插件实现word预览

首先, 我们先来创建一个Vue3项目 npm init vuelatest pnpm i npm run dev运行起来之后, 我们将App.vue中的代码全部删除掉 现在, 页面干净了, 我们需要安装vue-office插件 npm install vue-office/docx vue-demi安装完成之后, 我们就可以在页面中进行使用了 需要我们将组件…

边缘计算AI盒子目前支持的AI智能算法、视频智能分析算法有哪些,应用于大型厂矿安全生产风险管控

一、前端设备实现AI算法 主要是基于安卓的布控球实现&#xff0c;已有的算法包括&#xff1a; 1&#xff09;人脸&#xff1b;2&#xff09;车牌&#xff1b;3&#xff09;是否佩戴安全帽&#xff1b;4&#xff09;是否穿着工装&#xff1b; 可以支持定制开发 烟雾&#xf…

(免费分享)基于springboot,vue超市管理系统

开发工具&#xff1a;IDEA 服务器&#xff1a;Tomcat9.0&#xff0c; jdk1.8 项目构建&#xff1a;maven 数据库&#xff1a;mysql5.7 项目采用前后端分离 前端技术&#xff1a;vueelementUI 服务端技术&#xff1a;springbootmybatis-plusredis 本项目分为系统管理员、…

|行业洞察·手机|《2024手机行业及营销趋势报告-18页》

报告的主要内容解读&#xff1a; 手机行业概述及品牌分布&#xff1a; 2022年&#xff0c;受疫情影响&#xff0c;中国国内手机市场出货量下降22.6%&#xff0c;总计2.72亿部。5G手机市场占有率中&#xff0c;苹果领先&#xff0c;其次是vivo、OPPO和华为。消费者换机时更注重性…

鸿蒙OS开发实战:【悬浮窗口】

背景 悬浮视图或者窗体&#xff0c;在Android和iOS两大移动平台均有使用&#xff0c;HarmonyOS 也实现了此功能&#xff0c;如下为大家分享一下效果 准备 熟读HarmonyOS 悬浮窗口指导 熟读HarmonyOS 手势指导 熟读ALC签名指导&#xff0c;用于可以申请 “ohos.permission.S…

github | ssh拉取github仓库报错connect to host github.com port 22: Connection refused

配置ssh key 通过 ssh key 解决本地和服务器连接的问题 $ cd ~/. ssh #检查本机已存在的ssh密钥 如果提示 No such file or directory 则表示第一次使用git 输入&#xff1a; ssh-keygen -t rsa -C "邮件地址" 并且连续3次回车&#xff0c;最终会生成一个文件&am…

如何在Flutter中进行网络请求?

Hello&#xff01;大家好&#xff0c;我是咕噜铁蛋&#xff0c;你们的好朋友&#xff01;今天&#xff0c;我想和大家分享一下在Flutter中如何进行网络请求。Flutter作为一个跨平台的开发框架&#xff0c;网络请求是其实现数据交互的重要一环。下面&#xff0c;我将详细介绍几种…

JVM实战之性能调优[2](线程转储案例认识和分析)

文章目录 版权声明案例1&#xff1a;CPU占用率高问题问题描述解决思路补充内容 案例2&#xff1a;接口响应时间长问题问题描述解决思路Arthas trace命令Arthas watch命令解决问题 案例3&#xff1a;定位偏底层性能问题问题描述解决思路&#xff1a;Arthas火焰图问题解决 案例4&…

Siemens S7-1500TCPU 运动机构系统功能简介

目录 引言&#xff1a; 1.0 术语定义 2.0 基本知识 2.1 运动系统工艺对象 2.2 坐标系与标架 3.0 运动机构系统类型 3.1 直角坐标型 3.2 轮腿型 3.3 平面关节型 3.4 关节型 3.5 并联型 3.6 圆柱坐标型 3.7 三轴型 4.0 运动系统的运动 4.1 运动类型 4.1.1 线性运动…

ArcGIS Pro横向水平图例

终于知道ArcGIS Pro怎么调横向图例了&#xff01; 简单的像0一样 旋转&#xff0c;左转右转随便转 然后调整图例项间距就可以了&#xff0c;参数太多就随便试&#xff0c;总有一款适合你&#xff01; 要调整长度&#xff0c;就调整图例块的大小。完美&#xff01; 好不容易…

win10+cuda11.8+cudnn8.6.0安装

目录 一、NVIDIA 驱动程序下载 二、cuda11.8下载 三、cudnn8.6.0下载 四、确认cuda和cudnn是否安装成功 一、NVIDIA 驱动程序下载 1、查看显卡类型&#xff1a;连续按下CTRLALTDELETE -> 选择任务管理器 -> 性能 -> GPU -> 右上角 2、下载地址&#xff1a;官方…