二叉树OJ题详解

news2025/1/6 20:14:30

第一题:单值二叉树 力扣链接:力扣

7b4edfd6429446d08e54e0981a9ac5b6.png

 单值二叉树就是每一个节点存放的数据都相同,那么如何判断一棵树为单值二叉树呢?我们就拿最简单的一棵树为例子,比如根节点为1它的左子树和右子树也为1的一棵树,我们只需要比较根节点和它的左子树是否相等,根节点和它的右子树是否相等,也就是说判断一个二叉树是否为单值二叉树最重要的是判断父节点与左右子树的值是否相同。

/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     struct TreeNode *left;
 *     struct TreeNode *right;
 * };
 */


bool isUnivalTree(struct TreeNode* root){
    if (root==NULL)
    {
        return true;
    }
    if (root->left&&root->val!=root->left->val)
    {
        return false;
    }
    if (root->right&&root->val!=root->right->val)
    {
        return false;
    }
    return isUnivalTree(root->left)&&isUnivalTree(root->right);
}

 如代码我们可以看到首先判断这棵树是否为空,我们要知道一颗空数也是单值二叉树,所以当这棵树为空就返回true,然后去判断这个节点的左子树是否存在,如果存在就让这个节点与它的左子树相比,如果不等于就说明不是单值二叉树,当我们在判断一棵树是否是什么样的树的时候我们大部分都要去找反例。判断完左子树再去判断右子树,当两个if语句都没有返回时就去递归这棵树的左子树和右子树直到程序结束。 

第二题:二叉树最大深度 力扣链接:力扣 

57d17437f7cc49509547cafc1e59f3f6.png

 二叉树的最大深度相信我们并不陌生,我们在二叉树的构建中就讲解了二叉树的层数或者高度,这道题的本质就是求二叉树的高度。

/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     struct TreeNode *left;
 *     struct TreeNode *right;
 * };
 */


int maxDepth(struct TreeNode* root){
     if (root==NULL)
     {
         return 0;
     }
     int leftheight = maxDepth(root->left);
     int rightheight = maxDepth(root->right);
     return leftheight>rightheight?leftheight+1:rightheight+1;
}

 我们只需要注意,当这棵树为空,那么层数就为0。用两个变量去记录这棵树的左子树的高度和右子树的高度,我们只需要返回左右子树中最大的那个,当两个数相等时任意返回一个即可。所以我们在返回的时候用了三目操作符,当左子树高度大于右子树高度时,我们就返回左子树高度再加上根节点的那一层高度即可。

第三题:相同的树 力扣链接:力扣

37e68aae4ce5424cb006ec662d384c0e.png

怎么判断两棵树是否相同呢?其实很简单,只需要看着两棵树的节点是否相同并且节点的左右子树也都是相同的,从这个角度出发,那么当两个树都为空树的时候那么这两棵树一定是相同的,当一棵树为空一棵树不为空那么这两棵树一定不相同,然后再去递归这个节点的左子树和另一棵树的左子树是否相同同时右子树也必须相同。

/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     struct TreeNode *left;
 *     struct TreeNode *right;
 * };
 */


bool isSameTree(struct TreeNode* p, struct TreeNode* q){
       if (p==NULL&&q==NULL)
       {
           return true;
       }
       if (p==NULL||q==NULL)
       {
           return false;
       }
       if (p->val!=q->val)
       {
           return false;
       }
       return isSameTree(p->left,q->left)&&isSameTree(p->right,q->right);
}

 代码中的P 和 Q 就是两颗不同的树,当两棵树都为空时那么就返回true,当一棵树为空一棵树不为空的时候就返回false,当第一棵树的节点和第二棵树的节点不相同的时候就返回false,然后再去递归第一棵树的左子树和第二棵树的左子树是否相等并且第一棵树的右子树也要和第二棵树的右子树相等。

第四题:二叉树的前序遍历 力扣链接:力扣 

ddadc15a3464496cb59915ca8c7e7cc9.png

二叉树的前序遍历相信我们都并不陌生,但是这道题却有很多的坑,首先这道题并没有告诉我们数组的大小,也就是说我们要将遍历的数据放在数组中,但是每一个测试用例的数据多少都不一样,那么我们如何根据每一个测试用例有多少个节点就开辟多大的空间去存放数据呢?答案很简单,那就是先去计算这颗二叉树有多少个节点,然后直接给数组开辟这么大的空间即可。同样我们在往数组放数据的时候还需要一个下标,因为我们是用递归遍历的,所以这个下标一定要传地址过去。

/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     struct TreeNode *left;
 *     struct TreeNode *right;
 * };
 */


/**
 * Note: The returned array must be malloced, assume caller calls free().
 */

int TreeSize(struct TreeNode* root)
{
    if (root==NULL)
    {
        return 0;
    }
    return TreeSize(root->left)+TreeSize(root->right)+1;
}
void PrevOrder(struct TreeNode* root,int* a,int* i)
{
     if (root==NULL)
     {
         return;
     }
     a[(*i)++]=root->val;
     PrevOrder(root->left,a,i);
     PrevOrder(root->right,a,i);
}
int* preorderTraversal(struct TreeNode* root, int* returnSize){
      *returnSize = TreeSize(root);
      int*a = (struct TreeNode*)malloc(sizeof(int)*(*returnSize));
      int i = 0;
      PrevOrder(root,a,&i);
      return a;
}

 如代码所示,我们必须将此题划分三个函数,有人问全放在一个函数实现不行吗?答案是不可以,我们在计算节点大小和前序遍历都用的是递归,了解递归的本质就知道这是不可能实现的。计算二叉树的节点我们已经讲过,当节点为空就返回0,否则就返回这个节点的左子树的节点加上右子树的节点然后再加上自己,然后我们利用这个函数的返回值来给数组a开辟空间,空间开好后我们定义一个变量去记录下标,将这个树的根和数组和下标的地址都传给前序遍历函数,当有节点为空我们就return一下,否则就将值放到数组里,我们既然传的是地址所以需要先解应用再++,然后去递归这个节点的左子树,在这里可能会有人问为什么递归的函数里传的是i而不是地址呢?我们要明白递归是自己调用自己,自己本来一开始就接收的是地址,所以在递归函数中i就是第一次传来的地址,如果将i改成&i则相当于拿第一次传来的地址的地址,如果这样就要用二级指针,三级指针一直往后累加了,所以切记在递归中要明白传值的意义。当前序遍历完成后数组里已经放完了数据这时候直接返回a数组即可。

第五题:翻转二叉树 力扣链接:力扣

05fe4d25241442b383640770573af867.png

 什么是翻转二叉树呢?其实就是将一颗二叉树的左子树和右子树交换,然后这个左子树的左子树与这个左子树的右子树再交换,这样一看就是递归了,方法也很简单,每次都是保存左子树的节点的地址然后将右子树给左子树,然后再将保存的左子树给右子树,然后再递归到这个节点的左子树右子树。明白了这个我们看代码:

/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     struct TreeNode *left;
 *     struct TreeNode *right;
 * };
 */

void Reverse(struct TreeNode* root)
{
    if (root==NULL)
    {
        return;
    }
    struct TreeNode*tmp = root->left;
    root->left = root->right;
    root->right = tmp;
    Reverse(root->left);
    Reverse(root->right);
}
struct TreeNode* invertTree(struct TreeNode* root){
     if (root==NULL)
     {
         return NULL;
     }
     Reverse(root);
     return root;
}

首先我们先判断这棵树是否为空树,如果是空树的话直接返回空指针即可,如果不是就去翻转,进入翻转函数后我们用变量保存这棵树的左子树,然后将右子树给左子树然后把保存的左子树给右子树,在这里我们不要判断这棵树的左子树和右子树是否为空,因为为空也不影响交换并且还有一边为空一边不为空的情况,所以我们直接去递归它的左子树和右子树,在递归的过程中如果遇到空节点那么就return一下。全部翻转后返回这棵树即可。 

第六题:对称二叉树 力扣链接:力扣 

d6be82a976ab4507a52f7303c33c49b9.png

大家看到这道题有没有感觉到很熟悉呢?我们可以发现要判断一棵树是否对称,只需要判断这棵树的左子树的左子树是否等于这棵树的右子树的右子树,意思就是上图中左边2节点的左子树3要与右边2节点的右子树3相等。

/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     struct TreeNode *left;
 *     struct TreeNode *right;
 * };
 */
bool _isSymmetric(struct TreeNode* root1,struct TreeNode* root2)
{
   if (root1==NULL&&root2==NULL)
   {
       return true;
   }
   if (root1==NULL||root2==NULL)
   {
       return false;
   }
   if (root1->val!=root2->val)
   {
       return false;
   }
   return _isSymmetric(root1->left,root2->right)&&_isSymmetric(root1->right,root2->left);
}

bool isSymmetric(struct TreeNode* root){
    return !root||_isSymmetric(root->left,root->right);
}

我们在判断前需要先判断这棵树是否为空,如果为空则也是对称的然后去判断这棵树的左子树和右子树是否满足条件,当左子树为空并且右子树为空那么一定是对称的,当只有一边的树不为空那么一定是不对称的,当左节点不等于右节点也是不对称的。然后去递归这棵树的左子树的左子树与这棵树的右子树的右子树是否对称满足条件并且这棵树的左子树的右子树也要和这棵树的右子树的左子树满足条件。 

 第七题:另一棵树的子树 力扣链接:力扣

67bf043397324505b7b8e4cd647f3ef2.png

 在做了前面几道判断对称或者相同数的题后看这道题就非常简单了,其实就是在第一个树中找到与第二个数相同的节点结构。我们只需要在第一棵树中递归,当发现第一棵树中的子树与第二棵树相同,那么就返回true。

/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     struct TreeNode *left;
 *     struct TreeNode *right;
 * };
 */
bool isSumTree(struct TreeNode* root, struct TreeNode* subRoot)
{
    if (root==NULL&&subRoot==NULL)
    {
        return true;
    }
    if (root==NULL||subRoot==NULL)
    {
        return false;
    }
    if (root->val!=subRoot->val)
    {
        return false;
    }
    return isSumTree(root->left,subRoot->left)&&isSumTree(root->right,subRoot->right);
}

bool isSubtree(struct TreeNode* root, struct TreeNode* subRoot){
     if (root==NULL)
     {
         return false;
     }
     if (isSumTree(root,subRoot))
     {
         return true;
     }
     return isSubtree(root->left,subRoot)||isSubtree(root->right,subRoot);
}

 我们先判断第一棵树是否为空,如果第一棵树为空则一定不包含第二棵树。这里为什么不判断第二棵树为空呢?因为题目中告诉了第二棵树一定不为空。我们要写一个函数判断两棵树是否相同,当这个函数为真那么就返回true,否则就去递归第一棵树的左子树与第二棵树是否相等或者第一棵树的右子树与第二棵树是否相等。判断两棵树是否相同我们已经讲过,这道题只需要注意当在第一棵树的左子树发现了与第二棵树相同的数的时候直接返回true就不在递归第一棵树的右子树了,只有在第一棵树的左子树没有找到与第二棵树相同的数才需要去递归第一棵树的右子树。如果在第一棵树的左子树和右子树中都没有找到与第二棵树相同的树,那么就返回了false。 

第八题:二叉树的构建及遍历 牛客网链接:二叉树遍历_牛客题霸_牛客网 

43aa3b5f76f942088a3e36d998a700ac.png

二叉树的遍历这道题我们在二叉树的构建中已经讲过并且还画了递归图,我们就快速的讲解一下这道题的难点即可。

#include <stdio.h>
#include <stdlib.h>
struct TreeNode
{
    char val;
    struct TreeNode*left;
    struct TreeNode*right;
};
struct TreeNode* rebuildTree(char* str,int* i)
{
    if (str[*i]=='#')
    {
        (*i)++;
        return NULL;
    }
    struct TreeNode*newnode = (struct TreeNode*)malloc(sizeof(struct TreeNode));
    newnode->val = str[(*i)++];
    newnode->left = rebuildTree(str,i);
    newnode->right = rebuildTree(str,i);
    return newnode;
}
void middleOrder(struct TreeNode*root)
{
    if (root==NULL)
    {
        return;
    }
    middleOrder(root->left);
    printf("%c ",root->val);
    middleOrder(root->right);
}
int main() {
    char str[100];
    scanf("%s",str);
    int i = 0;
    struct TreeNode*tmp = rebuildTree(str,&i);
    middleOrder(tmp);
    return 0;
}

 首先,因为牛客网上的这道题是IO型,所以需要我们自己写主函数以及数的结构等,所以我们在主函数中先定义了一个char类型的数组,因为题目告诉字符串长度不会超过100,所以我们就将数组的大小定为100即可,然后输入这个字符串,并且还需要一个下标记录每次取到字符串中字符的位置,这点与前序遍历那道题一样都是需要传下标的地址,这棵树建好后传来的是这棵树的根节点,然后我们在用中序遍历打印这个数,在重建这棵树的函数中,当我们遇到字符“#”时,我们就要返回一个NULL让上一个节点链接,在这里要注意下标的++,当遇到正常字符的时候,我们就创建一个新节点,新节点的值就是数组中下标位置的字符,放完后直接让下标++指向下一个字符,然后让这个新节点的左子树去链接下一个字符这里是用递归的形式链接左右子树,当字符串走完后各个节点链接完成返回这棵树。中序遍历只需要注意先去递归左子树然后再打印值然后递归右子树即可。

 第九题:平衡二叉树 力扣链接:力扣

499451a2f9954a4daf7ff0c26bb28b85.png我们从题目介绍可以看到判断是否为平衡二叉树就看左右两个子树的高度差有没有超过1,我们从这个角度出发,用之前写过的求二叉树高度的函数,分别去求这棵树左子树的高度和右子树的高度,然后用两个变量去接收高度,一旦两个高度的差值大于1我们就返回false。否则就去递归这棵树的左子树是否满足平衡二叉树的要求并且这棵树的右子树也要满足平衡二叉树的要求。

int TreeHeight(struct TreeNode* root)
{
    if (root==NULL)
    {
        return 0;
    }
    int leftheight = TreeHeight(root->left);
    int rightheight = TreeHeight(root->right);
    return leftheight>rightheight?leftheight+1:rightheight+1;
}
bool isBalanced(struct TreeNode* root){
     if (root==NULL)
     {
         return true;
     }
     int left = TreeHeight(root->left);
     int right = TreeHeight(root->right);
     if (abs(left-right)>1)
     {
         return false;
     }
     return isBalanced(root->left)&&isBalanced(root->right);
}

求二叉树的深度我们已经讲过就不在多说,主要看平衡二叉树函数,当这棵树为空时那么也是平衡二叉树,然后去用变量分别记录这棵树的左子树和右子树的高度,abs是求绝对值,我们用这个函数求左右高度的绝对值,一旦出现大于1的情况直接返回false。如果没有提前返回那就递归这棵树的左子树,看这棵树的左子树是否满足平衡二叉树的条件并且右子树也要递归去判断是否满足条件,有一个没满足就返回false。

 

 

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

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

相关文章

以太网 TCP协议(TCP报文交互后的状态机变化)

2.7.2 以太网 TCP协议&#xff08;TCP报文交互后的状态机变化&#xff09; 一、TCP状态机&#xff1a; 二、TCP状态机变化 1、TCP三次握手 客户端主动发起SYN置位TCP之后&#xff0c;状态变为SYN_SENT(请求发送状态)服务器默认处于LISTEN(监听状态)。收到SYN报文之后&#x…

VMwareWorkStation如何添加万兆网卡,万兆网卡添加教程

1.引言 不同于ESXi&#xff0c;在VMware WorkStation&#xff08;后文简称VMware&#xff09;中添加网卡后没有选择网卡速度等级的选项&#xff0c;例如百兆、千兆、万兆等。就算点开右下角的”高级“也不管用。不过按照VMware的默认设定&#xff0c;当新建虚拟机选择32位操作系…

【玩转STL】STL的简介和string类用法和接口讲解(源码解析)

接触编程时间长一点的朋友想必都多多少少听过vector、string、queue等容器&#xff0c;也大抵了解一些有关STL的概念&#xff0c;这一节&#xff0c;我们就一起来谈一谈STL的六大组件&#xff0c;再来一起深入理解string类。 这里写目录标题&#x1f34e;、什么是STL&#xff0…

人工智能:语音合成技术介绍

❤️作者主页&#xff1a;IT技术分享社区 ❤️作者简介&#xff1a;大家好,我是IT技术分享社区的博主&#xff0c;从事C#、Java开发九年&#xff0c;对数据库、C#、Java、前端、运维、电脑技巧等经验丰富。 ❤️个人荣誉&#xff1a; 数据库领域优质创作者&#x1f3c6;&#x…

PCL 点云采样

一、简介 点云采样一般有三种方式&#xff0c;上采样&#xff0c;下采样&#xff0c;平均采样 原理介绍 下采样&#xff1a; 一般是采样是通过构建一个三维体素的格栅&#xff0c;然后在每个体素内用体素的重心近似的作为这个体素的整体特征&#xff0c;也就是说用这一个体素…

python 拆分pdf(有可执行文件exe)

1.背景 被那些软件pdf拆分整气死了&#xff0c;今天用python写一份pdf拆分的代码。 2.代码&#xff1a;&#xff08;计算机的可以去学习一下&#xff0c;自己改改&#xff09; pdf_split.py from PyPDF2 import PdfFileReader, PdfFileWriter# PDF文件分割 def split_pdf():…

pytorch nn.utils.rnn.pack_padded_sequence 分析

pack_padded_sequence 在nlp模型的forward方法中&#xff0c;可能有以下调用令读者疑惑 packed_embedded nn.utils.rnn.pack_padded_sequence(embedded, text_lengths, batch_firstTrue, enforce_sortedFalse)为什么要使用pack_padded_sequence&#xff1f; 参考 Pytorch中…

TDengine3.0:解决高基数问题的时序数据库设计思路

小 T 导读&#xff1a; 数据集的高基数&#xff08;High-Cardinality&#xff09;问题一直困扰着诸多主流的时序数据库&#xff08;Time Series Database&#xff0c;TSDB&#xff09;产品。一些数据库管理系统&#xff0c;在基数较低时表现良好&#xff1b;但是随着基数的增加…

vue2.x与vue3.x中自定义指令详解

&#x1f431;个人主页&#xff1a;不叫猫先生 &#x1f64b;‍♂️作者简介&#xff1a;前端领域新星创作者、华为云享专家、阿里云专家博主&#xff0c;专注于前端各领域技术&#xff0c;共同学习共同进步&#xff0c;一起加油呀&#xff01; &#x1f4ab;系列专栏&#xff…

m基于GA遗传算法的分件供送螺杆参数优化matlab仿真,优化参数包括螺杆总尺寸-最大圈数等

目录 1.算法描述 2.仿真效果预览 3.MATLAB核心程序 4.完整MATLAB 1.算法描述 首先介绍MATLAB部分的遗传算法的优化算法介绍&#xff1a; 遗传算法的原理 遗传算法GA把问题的解表示成“染色体”&#xff0c;在算法中也即是以二进制编码的串。并且&#xff0c;在执行遗传算法…

AI 对话模型被网友玩坏了!这次还可以运行 Docker 容器...

最近一款新的聊天 AI 被网友们玩疯了。它可以直接生成代码、可以给你的代码 debug 以及提出优化...可以模仿莎士比亚风格写作...还可以解答各种各样的问题&#xff0c;而且显然不只 10 岁小孩子的智商&#xff0c;感觉它已经把互联网上所有的公开资料都吸收并消化了。这就是 Op…

mssql(1433端口)介绍

mssql介绍 Microsoft SQL Server是一个关系型数据库&#xff0c;微软开发的管理系统。作为数据库服务器&#xff0c;它是一种软件产品&#xff0c;其主要功能是存储和检索其他软件应用程序所请求的数据&#xff0c;这些应用程序可以运行在同一台计算机上&#xff0c;也可以运行…

Java连接数据库(JDBC非常重要)

目录 一.数据库连接 1.1之前如何操作数据库 1.2.实际开发中如何操作数据库&#xff1f; 二.JDBC(Java Database Connectinity)(重要) 2.1.JDBC的概念 2.2 JDBC核心思想 2.2.1 MySQL数据库驱动 2.2.2 JDBC API 2.3JDBC 环境搭建 2.4准备一张表 2.4.1 创建student表 2.4.…

[附源码]Python计算机毕业设计Django三星小区车辆登记系统

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;…

Json简介与基本使用

前言 本文为Json简介与基本使用相关知识&#xff0c;下边具体将对什么是JSON&#xff0c;XML与JSON的区别&#xff0c;JSON的语法格式&#xff0c;JSON数据的转换&#xff08;包括&#xff1a;Java对象转换为JSON格式、JSON格式转换为Java对象&#xff09;等进行详尽介绍~ &am…

学习 MySQL:什么是分页

在本文中&#xff0c;我将解释在MySQL中&#xff0c;什么是分页以及我们如何实现它。当我们从 MySQL 数据库填充大型数据集时&#xff0c;读取同一页面上的所有记录并不容易。使用分页&#xff0c;我们可以将结果集划分为多个页面&#xff0c;这增加了结果集的可读性。在本文中…

协议-序列化-http-Cookie-Session-https

文章目录再谈协议什么是序列化&#xff0c;什么是反序列化&#xff1f;为什么要进行序列化和反序列化&#xff1f;怎么进行序列化和反序列化呢&#xff1f;网络计算器版本总结http协议httpurlencode&urldecode一. 格式认识二 代码实现一个http协议下的服务器安装telnet服务…

javaScript 进阶之路 --- 《加深理解回调函数》

前言&#xff1a; 回想当初第一次看到“回调函数”这个名词的时候&#xff0c;真的快把我难哭了。所有视频教程在讲到某个知识点的时候&#xff0c;大概都会说一句&#xff1a;“啊&#xff0c;这里怎么办呢&#xff1f;这里我们就需要用到一个回调函数...”。 等等&#xff0…

校园论坛(Java)—— 数据报表模块

校园论坛&#xff08;Java&#xff09;—— 数据报表模块 文章目录校园论坛&#xff08;Java&#xff09;—— 数据报表模块1、写在前面2、系统结构设计2.1 各个页面之间的调用关系2.2.3、数据报表设计3.1 数据报表主界面的实现3.2 发表数Top5的普通帖子3.3 回帖数Top5的普通帖…

技术人员创业的第一步分析(续,可听音频)

概述&#xff1a;昨天的文章发布以后&#xff0c;在腾讯云TVP专家群里和多个技术群里都引起了一些讨论&#xff0c;基于这些讨论&#xff0c;有了今天的这篇续章。里面谈到了这次创业中&#xff0c;青润经历过的几次生死关头&#xff0c;是真的差点离开人世&#xff0c;而不是想…