C++——二叉树OJ

news2024/11/16 15:27:10

目录

1.根据二叉树创建字符串

2.二叉树的层序遍历  

3.二叉树的层序遍历2 

4.二叉树的最近公共祖先 

5. 搜索二叉树与双向链表

 6.从前序与中序遍历构建二叉树


 

1.根据二叉树创建字符串

按照前序遍历:根左右。

1(2(4()())())(3()())

 1.左右都为空,则可以省略掉括号

如这里的3,还有4 1(2(4)())(3)

2.左子树不为空,右子树为空,省略掉

如2,1(2(4))(3)

3.如果左为空。不省略

class Solution
 {
public:
    string tree2str(TreeNode* root)
     {
if(root==nullptr)
return string();
string str;
str+=to_string(root->val);//根节点

if(root->left||root->right) //左边不为空或者左边为空右边不为空,左边需要加括号
{
str+='(';                 //左子树
str+=tree2str(root->left);
str+=')';
}

if(root->right)//右边不为空,则加括号
{
str+='(';                 //右子树
str+=tree2str(root->right);
str+=')';
}
return str;
    }
};

2.二叉树的层序遍历  

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

思路:使用队列,当某个节点出来的时候,将他的子节点放入

 

 

9出来后放入30,但有个问题,不知道每一层的分界点在哪 ,这里当20出来后不知道30是这一场的还是下一层的

解决办法:用一个数字去记录每层的个数,控制一层一层出

class Solution {
public:
    vector<vector<int>> levelOrder(TreeNode* root) {
 queue<TreeNode*> q;
 size_t levelSize=0;
 if(root)
 {
     q.push(root);
     levelSize=1;
 }
 vector<vector<int>> vv;
 while(!q.empty())
 {
     vector<int> v;
     for(size_t i=0;i<levelSize;++i)
     {
         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);
     }
     vv.push_back(v);//把这一层给v
     levelSize=q.size();
 }
 return vv;
    }
};

3.二叉树的层序遍历2 

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

逆置一下即可

class Solution {
public:
    vector<vector<int>> levelOrderBottom(TreeNode* root) {

 queue<TreeNode*> q;
 size_t levelSize=0;
 if(root)
 {
     q.push(root);
     levelSize=1;
 }
 vector<vector<int>> vv;
 while(!q.empty())
 {
     vector<int> v;
     for(size_t i=0;i<levelSize;++i)
     {
         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);
     }
     vv.push_back(v);//把这一层给v
     levelSize=q.size();
 }
 reverse(vv.begin(),vv.end());
 return vv;
    }
    };

4.二叉树的最近公共祖先 

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

 思路:如果一个是某节点左子树中的节点,一个是它右子树中的节点,该节点就是公共祖先

  //1.一个在左,一个在右,root是公共祖先
    //2.都在左,递归到左子树找
    //3.都在右,递归到右子树找

class Solution {
public:
bool Find(TreeNode *sub,TreeNode *x)
{
  if(sub==nullptr)
  return false;
  if(sub==x)
  return true;
  return sub==x||Find(sub->left,x)
           ||Find(sub->right,x);
}
    TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
          if(root==nullptr)
    return nullptr;
    if(root==p||root==q)//如果p或q是根节点,就说明root是公共祖先
    {
        return root;
    }
    bool pInleft,pInright,qInleft,qInright;
    pInleft=Find(root->left,p);//在左子树中去找p
    pInright=!pInleft;
        qInleft=Find(root->left,q);//在右子树中去找q
    qInright=!qInleft;
    //1.一个在左,一个在右,root是公共祖先
    //2.都在左,递归到左子树找
    //3.都在右,递归到右子树找
    if((pInleft&&qInright)||(qInleft&&pInright))
    return root;
    else if(pInleft&&qInleft)//如果都在左
    {
return lowestCommonAncestor(root->left,p,q);
    }
        else if(pInright&&qInright)//如果都在左
    {
return lowestCommonAncestor(root->right,p,q);
    }
    else  
    return nullptr;
    }
};

时间复杂度是O(H*N),H是树的高度,我们将时间复杂度优化到O(N)

使用一个栈解决,如果不是我们要找的节点就入栈,如这里要找6,4把往6这条路走的节点都入栈。之后返回。

 这里找4,先找左数3,5,6 6的左右子树为空,就说明没找到4,把6出栈

.出栈之后从2开始找,从左子树开始找,再找右子树

 

 此时有俩个栈

 

 我们让大的栈先pop,先pop4,然后比较2和6不相等pop2和6,再比较5和5相等就说明找到了

class Solution {
public:
bool FindPath(TreeNode *root,TreeNode* x, stack<TreeNode*>&path)
{
 if(root==nullptr)
 return false;
 path.push(root);//先入栈
 if(root==x)
 return true;//找到了该节点
 if(FindPath(root->left,x,path))//如果没找到就去左数找
 return true;//左树找到了就返回
 if(FindPath(root->right,x,path))//左树没找到就去右树找
 return true;
 path.pop();//左边右边都没找到pop出去该节点
 return false;//左右都没找到
}
TreeNode *lowestCommonAncestor(TreeNode *root,TreeNode *p,TreeNode *q)
{
    stack<TreeNode*>pPath,qPath;
    FindPath(root,p,pPath);
    FindPath(root,q,qPath);
    //类似链表相交,长的先走
    while(pPath.size()!=qPath.size())
    {
        if(pPath.size()>qPath.size())
        pPath.pop();
        else
        qPath.pop();
    }
    while(pPath.top()!=qPath.top())
    {
        pPath.pop();
        qPath.pop();
    }
    return pPath.top();
}
};

5. 搜索二叉树与双向链表

二叉搜索树与双向链表_牛客题霸_牛客网 (nowcoder.com)

 中序递归遍历,这里让left指向中序顺序的前一个,right指向中序顺序的下一个

用俩个指针prev和cur,cur比prev先走一步。

走到这时让cur->left指向prev

由于是中序遍历,cur走到6之后会走到8,让8的left指向6

 解决了left,接下来解决right,right要指向下一个即4->6->8->10

 换个遍历顺序:右->根->左这样可以处理右节点

还下面这种办法,让prev的右指向6

 让6的右指向8

class Solution {
public:
void InOderConvert(TreeNode*cur,TreeNode*& prev)
{
    if(cur==nullptr)
	return ;
	 InOderConvert(cur->left,prev);
	 cur->left=prev;
	 if(prev)
	 prev->right=cur;
	 prev=cur;
	  InOderConvert(cur->right,prev);

}
    TreeNode* Convert(TreeNode* pRootOfTree) {
		TreeNode* prev=nullptr;
        InOderConvert(pRootOfTree,prev);
		TreeNode* head=pRootOfTree;
		while(head&&head->left)
		{
			head=head->left;
		}
		return head;
    }
};

最后要返回链表头部,根据根节点往左边找即可找到头部,prev要用引用,是因为递归返回上一层的时候要把确定新的prev的位置,不然prev在第一层是空,第二层是6,当第二成返回第一层时,cur->left=prev,此时应该是10->6,而这里不加引用就是10->nulllptr

 6.从前序与中序遍历构建二叉树

105. 从前序与中序遍历序列构造二叉树 - 力扣(LeetCode)

 大思路:前序创建树,中序分割左右子树

前序可以确定根

class Solution {
public:
 TreeNode* _buildTree(vector<int>& preorder, vector<int>& inorder,int& prei,int inbegin,int inend)
{
    if(inbegin>inend)
    return nullptr;
TreeNode *root=new TreeNode(preorder[prei++]);//分割中序
int ini=inbegin;
while(ini<=inend)
{
    if(inorder[ini]==root->val)
    break;
    else
    ++ini;
}//创建根
 //[inbegin,ini-1]ini [ini,inend]
root->left=_buildTree(preorder,inorder,prei,inbegin,ini-1);
root->right=_buildTree(preorder,inorder,prei,ini+1,inend);
return root;
 }
    TreeNode* buildTree(vector<int>& preorder, vector<int>& inorder) {
int i=0;
 return _buildTree(preorder,inorder,i,0,inorder.size()-1);
    }
};

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

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

相关文章

通过PWM控制串行LED灯

资料下载 RGB_LED灯带_5050慢闪_datasheet STM32控制LED灯带 根据上面的说明书可知&#xff0c;通过修改800KHz的PWM波形的占空比可以控制LED的颜色。 假设现在有3颗串联起来的灯珠&#xff0c;如下图&#xff1a; 如果U1/U2/U3需要显示红/绿/蓝色&#xff0c;根据说明书&…

Vue3 的项目搭建

使用工具&#xff1a; VSCode Volar &#xff1a;编辑器和插件Vite&#xff1a;打包工具Vue3&#xff1a;VueX&#xff1a;存储NodeJs&#xff1a;环境 操作步骤&#xff1a; 安装Node.js安装VsCode安装 Volar创建一个Vite的初始化项目 npm init viteProject name 这一行 我…

Linux系统下刨析hello world背后的秘密

Hi&#xff0c;小伙伴们&#xff0c;大家好&#xff01;今天给大家讲解Linux系统编程中几个重要的概念。必须理解了这几个重要概念&#xff0c;才能更快的入门Linux系统编程&#xff0c;这是不可或缺的基础知识。看似简单&#xff0c;如果不花一番功夫很难真正的理解它们。需要…

113.(leaflet之家)leaflet根据距离截取线段

听老人家说:多看美女会长寿 地图之家总目录(订阅之前建议先查看该博客) 文章末尾处提供保证可运行完整代码包,运行如有问题,可“私信”博主。 效果如下所示: 下面献上完整代码,代码重要位置会做相应解释 <!DOCTYPE html> <html>

AAC ADTS格式分析与提取aac音频文件实战

1.AAC⾳频格式简介 AAC⾳频格式&#xff1a;Advanced Audio Coding(⾼级⾳频解码)&#xff0c;是⼀种由MPEG-4 标准定义的有损⾳频压缩格式&#xff0c;由Fraunhofer发展&#xff0c;Dolby, Sony和AT&T是主 要的贡献者。 AAC 包含两种格式&#xff1a;ADIF和ADTS 2.ADIF…

Python基础(二十):面向对象基础

文章目录 面向对象基础 一、理解面向对象 二、类和对象 1、理解类和对象

已知Retangle2的中心点坐标,角度,半长,半宽,计算四个顶点坐标和四个线段中点坐标

Halcon可以通过最小外接矩型算子获取Retangle2的中心点坐标&#xff0c;角度&#xff0c;半长&#xff0c;半宽。但是却没有四个顶点坐标和四个线段中点坐标。下面的代码是获取四个顶点坐标和四个线段中点坐标的方法。 gen_rectangle2 (Rectangle, 300, 200, 0.39, 100, 50) C…

E5071C网络分析仪

18320918653 E5071C 安捷伦E5071CAgilent 8.5G网络分析仪二手E5071C 频率范围&#xff1a;300 kHz至8.5GHz 在测试端口处保持125 dB动态范围&#xff08;典型值&#xff09; 宽动态范围&#xff1a;在测试端口上的动态范围> 123 dB&#xff08;典型值&#xff09; 极快的…

GDB调试技巧实战--找到打开某个文件的地方

问题&#xff1a; 本系列期望用一个简单的实例来展示一些调试技巧。不时更新。 本期问题是&#xff1a;如何才能快速定位到打开某个文件的代码&#xff1f;&#xff08;对应open/fopen, 读者可举一反三到其它系统函数&#xff09;。当面对屎山代码时&#xff0c;GDB的条件断点…

基于R的Bilibili视频数据建模及分析——变量相关性分析篇

基于R的Bilibili视频数据建模及分析——变量相关性分析篇 文章目录基于R的Bilibili视频数据建模及分析——变量相关性分析篇0、写在前面1、数据分析1.1 变量相关性分析1.2 单元数据直观展示1.3 多元数据直观展示2、多元数据与回归分析2.1 简单相关分析2.2 简单线性回归分析2.3 …

从0到1搭建大数据平台之调度系统

大家好&#xff0c;我是脚丫先生 (o^^o) 大数据平台核心之一在于数据计算&#xff0c;分为离线计算和实时计算任务。 然而任务是离不开调度的。比如&#xff1a;我们要进行定时抽取业务数据库 的数据&#xff0c;定时跑hive/spark任务&#xff0c;定时推送日报、月报指标数据…

数字档案馆系统测试前准备工作(“指标表”和“具体要求”可下载)

数字档案馆系统测试是国家档案局组织开展的一项针对县级以上国家综合档案馆数字档案馆建设情况的绩效评价工作。数字档案馆系统测试的评分依据是《数字档案馆系统测试指标表》&#xff08;后台回复“指标表”可下载&#xff09;&#xff0c;测试采用百分制。测试结果达到80分以…

华脉智联电力行业技术解决方案

一、前言 所谓电力应急&#xff0c;就是快速处理突发紧急事件尽量减少因供电中断造成的损失&#xff0c;快速修复故障设备或线路&#xff0c;使电力生产和运营恢复正常。重大施工事故、公共紧急事件、性质恶劣的违章操作、蓄意破坏、自然灾害等&#xff0c;都极易引发大规模停…

论文浅尝 | Future Event Prediction Based on Temporal KG Embedding

笔记整理&#xff1a;杨露露&#xff0c;天津大学硕士链接&#xff1a;https://www.techscience.com/csse/v44n3/49146/html动机对未来事件的准确预测在许多领域为社会带来巨大利益&#xff0c;减少损失&#xff0c;如内乱、流行病和犯罪。知识图谱是描述和建模复杂系统的通用语…

【状态估计】将变压器和LSTM与卡尔曼滤波器结合到EM算法中进行状态估计(Python代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

讨教大学|六西格玛之属性值数据一致性分析

相信大家在日常的工作中肯定会遇到这样一种情况&#xff1a;产品的质量特性属于外观特性&#xff0c;即 需要靠人工判断产品是不是符合标准。当遇到客户投诉&#xff0c;或者内部报废率比较高的时候&#xff0c;经 常会遇到客户和领导的挑战&#xff1a; 如何保证员工清楚的知…

IO、NIO、BIO傻傻分不清吗,让我对象告诉你~~

1、Stream 与 Channel stream 不会自动缓冲数据&#xff0c;channel 会利用系统提供的发送缓冲区、接收缓冲区&#xff08;更为底层&#xff09;stream 仅支持阻塞 API&#xff0c;channel 同时支持阻塞、非阻塞 API&#xff0c;网络 channel 可配合 selector 实现多路复用二者…

电商系统概要设计

设计一个最小化的电商系统&#xff0c;理清楚电商系统的架构。目前唯一能确定的是&#xff0c;老板要做一个电商系统。具体做成什么样&#xff0c;还不清楚。你需要和老板讨论需求。 你&#xff1a;“咱们要做的业务模式是C2C、B2C还是B2B呢&#xff1f;” 老板&#xff1a;“…

CES 2023:NVIDIA发力元宇宙与汽车市场

作为全球规模最大、影响最为广泛的国际消费电子展&#xff0c;2023年的CES&#xff08;Consumer Electronics Show&#xff09;在美国拉斯维加斯如约而至。众所周知&#xff0c;作为全球消费电子产业发展的风向标&#xff0c;每年的CES都汇聚了各大科技企业最先进的技术与产品。…

Renesas setting-文件导入

导入工程 1.1, 文件\从文件系统选中打开项目… 1.2, 打开xpg_app文件 链接文件 2.1&#xff0c;选中项目\右键\属性 2.2&#xff0c;删除文件&#xff08;在不同电脑中绝对路径不同&#xff09; 属性\路径和符号&#xff0c; 源位置&#xff08;delete C 文件&#xff09;&…