二叉树——经典练习题

目录

前言:

一、单值二叉树

        题目描述:

        思路分析:

        代码实现:

二、二叉树最大深度 

        题目描述:

        思路分析:

                代码实现: 

三、检查两颗树是否相同

        题目描述:

        思路分析: 

        代码实现:

四、另一颗树的子树

        题目描述:

        思路分析: 

        代码实现: 

五、二叉树的前序遍历

        题目描述:

        思路分析:

        代码实现:

六、二叉树的构建及遍历

        题目描述:

        思路分析: 

        代码实现:

最后:


前言:

        通过前面的学习,我们已经学习了二叉树的基础知识,以及二叉树功能的实现。我们学习讲究的是:知行合一。那我们一起来检验一下我们的学习成果吧!

一、单值二叉树

        题目描述:

如果二叉树每个节点都具有相同的值,那么该二叉树就是单值二叉树。

只有给定的树是单值二叉树时,才返回 true;否则返回 false

        思路分析:

                单值二叉树就是二叉树每个节点的值都相同,这时,我们有两种思路去实现:

                方法一:再构造一个函数,把根节点的值传过去和左节点和右节点进行比较,如果相同,这个没啥意义,应用不同来判断,如果不同就返回假,为真就继续递归。

                方法二:用根节点和其左右节点进行比较,思路与上文类似。

                本题采用方法二。

        代码实现:

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

                以下是递归展开图,大致展示了函数调用过程,帮助大家理解:

                题目链接:. - 力扣(LeetCode) 

二、二叉树最大深度 

        题目描述:

给定一个二叉树 root ,返回其最大深度。

二叉树的 最大深度 是指从根节点到最远叶子节点的最长路径上的节点数。

        思路分析:

                此题一看不是和之前讲解的高度一样吗?对于我之前的说明不以为然的读者,可能会写出以下代码:

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

                这段代码运行的大逻辑是没错的,但是会有效率问题,测试用例通过是没啥问题的,但要是要求时间复杂度了,就危险了。

                可以看出,超出了时间限制,这时我们就要对其进行优化,具体过程上一篇文章已经分析过了,优化方法为:加一个记录的变量即可,便可减少复杂度。

                代码实现: 

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

                题目链接:. - 力扣(LeetCode)

三、检查两颗树是否相同

        题目描述:

给你两棵二叉树的根节点 p 和 q ,编写一个函数来检验这两棵树是否相同。

如果两个树在结构上相同,并且节点具有相同的值,则认为它们是相同的。

        思路分析: 

                本题要求判断两颗树是否相同,那么,这么说明两颗树相同呢?无外乎就是一一比较,根与根比较,左子树与左子树比较,右子树与右子树比较,就这么简单。但,我们还要考虑的情况是:如果为空,这么判断,分两种情况:一、都为空:那就返回true。二、一个为空,另一个不为空。该放回false。以上便是所有情况。

        代码实现:

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);
}

        题目链接:. - 力扣(LeetCode)

四、另一颗树的子树

        题目描述:

给你两棵二叉树 root 和 subRoot 。检验 root 中是否包含和 subRoot 具有相同结构和节点值的子树。如果存在,返回 true ;否则,返回 false 。

二叉树 tree 的一棵子树包括 tree 的某个节点和这个节点的所有后代节点。tree 也可以看做它自身的一棵子树。

        思路分析: 

                本题一看没啥思路,这咋搞,很难办对吧。这时,突然想到,能不能放到数组里,然后看小数组是不是大数组的子集。但是,画图分析一下又好像不行。

                唯一的思路断了,这该怎么办?在思索之时,你突然想到:专业的事交给专业的人去干。这道题可不可以分解为这两步:第一步:先找到所有父节点。第二步:在经行判断两棵树是否相同。对于第二步和上题有异曲同工之妙,简单优化即可。

                第一步应该怎么办?很简单,直接把节点扔进去就行了,交给第二步去判断就行了,直到找到符合的节点。

                所以,这道题看起来很复杂,其实也不简单,对吧。

        代码实现: 

bool istree(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 istree(root->left,subRoot->left) && istree(root->right , subRoot->right);
}

bool isSubtree(struct TreeNode* root, struct TreeNode* subRoot){
    if(root == NULL)
    {
        return false;
    }
    if(root->val == subRoot->val && istree(root,subRoot))
    {
        return true;
    }
    return isSubtree(root->left,subRoot) || isSubtree(root->right , subRoot);//找父节点
}

        题目链接:. - 力扣(LeetCode)

五、二叉树的前序遍历

        题目描述:

        给你二叉树的根节点 root ,返回它节点值的 前序 遍历。

        思路分析:

                二叉树的前序遍历,这不是说过吗?那好,给大家这个接口,看一看能否在leetcode官方上能不能运行通过。

int* preorderTraversal(struct TreeNode* root, int* returnSize) {
    
}

                对于以上接口这里解释一下返回值这个问题。一个函数不是只有一个,那咱们应该返回啥,对于该问题,我们要先明白传过来的都代表什么意思。

                咱们首先明白,只要实现接口里的大逻辑即可,输入和打印咱们是不必实现的。打印是有leetcode官方写的代码实现的,你要实现打印肯定要知道这个树的的大小对吧。所以,那个returnsize是用来记录大小的。那为什么不用常量呢?因为:形参是实参的一份临时拷贝。要改变形参要传地址,地址需要用指针来接收。

                说了这么多,如果读者能够实现,就不用看博主的讲解,如果不行,就一件三连跟着博主继续走即可。

                对于我们现在来说,求一个树的节点个数是很简单的一件事,就不在赘述。知道了节点数,便可动态开辟一个数组用来存放数据。那么,我们接下来的难点为:如何把二叉树中的元素放到数组中。

                这件事也很简单,就是用前序遍历,把打印换成放元素仅此而已。

        代码实现:

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

        题目链接:. - 力扣(LeetCode)

六、二叉树的构建及遍历

        题目描述:

描述

编一个程序,读入用户输入的一串先序遍历字符串,根据此字符串建立一个二叉树(以指针方式存储)。 例如如下的先序遍历字符串: ABC##DE#G##F### 其中“#”表示的是空格,空格字符代表空树。建立起此二叉树以后,再对二叉树进行中序遍历,输出遍历结果。

输入描述:

输入包括1行字符串,长度不超过100。

输出描述:

可能有多组测试数据,对于每组数据, 输出将输入字符串建立二叉树后中序遍历的序列,每个字符后面都有一个空格。 每个输出结果占一行。

         

        思路分析: 

                本题要求我们实现一个二叉树,并用中序遍历的方式将其打印出来。这道题很有难度,但有了前面的代码,在此时难度也不是很大了。本题的数据从外部输入,我们还要将其打印。所以,我们要自己实现接受外部的输入和用中序的方式对外打印,这部分代码难度不大,中序打印的类似实现已经讲过,不在赘述。

                本题讲解的难点为:构建二叉树。从以上的输入示例中,我们看到有‘#’这样的符号,但打印没它,那,我们可不可以写出以下代码:

if(a[(*pi)++] == '#')
    {
        return NULL;
    }

                乍一看没啥问题。但仔细想想问题就来了:这个的意思是我每次判断pi都加一,你觉得可行还是不可行。很显然,这不可行。所以,我们要把pi的++放入到{}内才能更好的符合我们的意图。我们放入字符采用的是先序遍历,这部分大家都能掌握,只不过,我们不清楚整颗二叉树的大小,所以,我们每次构建前都要用malloc开辟一下。

        代码实现:

#include <stdio.h>
#include<stdlib.h>
typedef char BTDataType;

typedef struct BinaryTreeNode {
    BTDataType data;
    struct BinaryTreeNode* left;
    struct BinaryTreeNode* right;
} BTNode;
BTNode* BTCreatTree(char* a,int* pi)
{
    if(a[*pi] == '#')
    {
        (*pi)++;
        return NULL;
    }
    BTNode* root = (BTNode*)malloc(sizeof(BTNode));
    root->data = a[(*pi)++];
    root->left = BTCreatTree(a,pi);
    root->right = BTCreatTree(a,pi);
    return root;
}
void BTNodePrint(BTNode* root)
{
    if(root == NULL)
    {
        return;
    }
    BTNodePrint(root->left);
    printf("%c ",root->data);
    BTNodePrint(root->right);
}
int main() {
   char a[101];
   scanf("%s",a);
   int i = 0;
   BTNode* ret = BTCreatTree(a,&i);
   BTNodePrint(ret);
    return 0;
}

        题目链接:二叉树遍历_牛客题霸_牛客网

最后:

        二叉树的学习,我们目前就告一段落了,后续的进阶内容会在c++部分讲解。今天讲解的题目中最后三道难度较大,还望各位读者在学习完后能够多多练习,这样才能够掌握。如在学习中,有啥问题可在评论区交流,也可私信。期待与读者再会。

完!

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

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

相关文章

AI图书推荐:ChatGPT解码—人工智能增强生活指南

《ChatGPT解码—人工智能增强生活指南》&#xff08;ChatGPT Decoded. A Beginners Guide to AI-Enhanced Living &#xff09;是一本由 大卫维恩斯&#xff08;David Wiens &#xff09;所著的书籍&#xff0c;旨在帮助读者了解并有效利用GPT-4语言模型这一强大工具来提升日常…

OceanBase SQL 诊断和调优实践——【DBA从入门到实践】第七期

数据库作为绝大多数应用系统储存数据的核心系统&#xff0c;在用户系统需要访问数据时&#xff0c;有着至关重要的作用。在这些交互中&#xff0c;SQL 语言是应用与数据库系统之间“沟通”的桥梁&#xff0c;它负责将应用的指令传达给数据库。因此&#xff0c;SQL 的性能好坏直…

网络工程师---第三十八天

ISIS&#xff1a; ISIS含义&#xff1a;中间系统到中间系统IS-IS。 ISIS特点&#xff1a;①内部网关协议IGP&#xff08;Interior Gateway Protocol&#xff09;&#xff0c;用于自治系统内部&#xff1b; ②IS-IS也是一种链路状态协议&#xff0c;使用最短路径优先SPF算法进…

Jenkins 流水线(Pipeline)详解

大家好&#xff0c;Jenkins 流水线&#xff08;Pipeline&#xff09;是一种可编排的持续集成和交付&#xff08;CI/CD&#xff09;方法&#xff0c;它以代码的方式定义整个软件开发过程中的构建、测试和部署流程。接下来就跟大家分享一下Jenkins 流水线&#xff08;Pipeline&am…

云计算-基础设施和管理机制(Infrastructure and Management Mechanisms)

逻辑网络边界&#xff08;Logical Network Perimeter&#xff09; 逻辑网络边界是软件控制的虚拟网络&#xff0c;它是物理网络的一部分。其主要思想是隔离逻辑网络&#xff0c;防止不希望的访问&#xff0c;同时仍然为合法用户提供访问权限。下图显示了云系统中一个简单的逻辑…

【Qt】Qt入门

思维导图 学习目标 这一系列是学习Qt&#xff0c;在C中&#xff0c;会发现有不少岗位的要求是熟悉Qt&#xff0c;所以Qt的学习是不能推迟的。 一、Qt的概述 1.1 Qt的特点 Qt是一个跨平台的C应用程序开发框架&#xff1a; 具有短平快的优秀特质&#xff1a;投资少&#xff0…

每日练习——同余方程以及格雷码

同余方程 题目描述 运行代码 #include<iostream> #define ll long long using namespace std; ll exgcd(ll a, ll b, ll& x, ll& y) {if (!b)return x 1, y 0, a;ll d exgcd(b, a % b, y, x);y - a / b * x;return d; } int main() {ll a, b, x, y;cin >…

【教学类-58-05】黑白三角拼图05(2-10宫格,每个宫格随机1张-6张,带空格纸,1页3张黑白3张白卡)

背景需求&#xff1a; 【教学类-58-04】黑白三角拼图04&#xff08;2-10宫格&#xff0c;每个宫格随机1张-6张&#xff0c;带空格纸&#xff0c;1页6张黑白&#xff0c;1张6张白卡&#xff09;-CSDN博客文章浏览阅读582次&#xff0c;点赞16次&#xff0c;收藏3次。【教学类-58…

Kafka 安装教程和基本操作

一、简介 Kafka 是最初由 Linkedin 公司开发&#xff0c;是一个分布式、分区的、多副本的、多订阅者&#xff0c;基于 zookeeper 协调的分布式日志系统&#xff08;也可以当做 MQ 系统&#xff09;&#xff0c;常见可以用于 web/nginx 日志、访问日志&#xff0c;消息服务等等…

C++之lambda函数与std::bind区别及用法实例(二百八十)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a;多媒…

AI网络爬虫-从当当网批量获取图书信息

工作任务和目标&#xff1a;用户输入一个图书名称&#xff0c;然后程序自动从当当网批量获取图书信息 查看相关元素在源代码中的位置&#xff1a; 第一步&#xff1a;在deepseek中输入提示词&#xff1a; 你是一个Python爬虫专家&#xff0c;一步步的思考&#xff0c;完成以下…

5.26机器人基础-DH参数 正解

1.建立DH坐标系 1.确定Zi轴&#xff08;关节轴&#xff09; 2.确定基础坐标系 3.确定Xi方向&#xff08;垂直于zi和zi1的平面&#xff09; 4.完全确定各个坐标系 例子&#xff1a; 坐标系的布局是由个人决定的&#xff0c;可以有不同的选择 标准坐标系布局&#xff1a; …

HAL库LED点灯

一、搭建开发环境 &#xff08;一&#xff09;安装MDK5 具体安装请参照下面链接&#xff1a;如何开始一个stm32的简单程序的编译_stm32程序编译-CSDN博客 &#xff08;二&#xff09;安装Jdk 由于STM32CubeMX需要用到JAVA&#xff0c;因此需要安装jdk环境。 jdk官网下载链接…

素数判断的奥秘与编程实践

新书上架~&#x1f447;全国包邮奥~ python实用小工具开发教程http://pythontoolsteach.com/3 欢迎关注我&#x1f446;&#xff0c;收藏下次不迷路┗|&#xff40;O′|┛ 嗷~~ 目录 一、素数定义的深入理解 二、非素数的例子与思考 三、素数判断的编程实现 1. 穷举法判断素…

protobuf —— 认识和安装

protobuf —— 认识和安装 什么是序列化和反序列化有哪些常见的什么是序列化和反序列化工具Protobuf安装安装依赖开始安装 连接动态库一些遗留问题 我们今天来看一个序列化和反序列化的工具&#xff1a;protobuf。 什么是序列化和反序列化 序列化&#xff08;Serialization&a…

基于SpringBoot和Mybatis实现的留言板案例

目录 一、需求及界面展示 二、准备工作 引入依赖 .yml文件相关配置 数据库数据准备 三、编写后端代码 需求分析 代码结构 Model Mapper Service Controller 前端代码 四、测试 一、需求及界面展示 需求&#xff1a; 1. 输入留言信息&#xff0c;点击提交&…

MySQL:图文超详细教程MySQL5.7下载与安装

一、前言 MySQL 5.7 是一个重要的数据库管理系统版本&#xff0c;它带来了多项改进和新特性&#xff0c;本文将超详细的带大家手动安装一下MySQL5.7。 二、下载MySQL5.7版本 MySQL5.7安装包 链接&#xff1a;https://pan.baidu.com/s/1lz5rp9PwfyeHzkEfI_lW6A 提取码&#…

MyBatis框架的使用:mybatis介绍+环境搭建+基础sql的使用+如何使用Map传入多个参数+返回多个实体用List或者Map接收+特殊sql的使用

MyBatis框架的使用&#xff1a;mybatis介绍环境搭建基础sql的使用如何使用Map传入多个参数返回多个实体用List或者Map接收特殊sql的使用 一、MyBatis介绍1.1 特性1.2 下载地址1.3 和其它持久层技术对比 二、搭建环境2.1配置maven2.2 创建mybatis配置文件2.3 搭建测试环境 三、基…

spring状态机实战

引言 完整代码库gitee地址:代码地址 一、什么是状态机 状态机是有限状态自动机的简称&#xff0c;是现实事物运行规则抽象而成的一个数学模型&#xff0c;是一种概念性机器&#xff0c;它能采取某种操作来响应一个外部事件。这种操作不仅能取决于接收到的事件&#xff0c;还…

如何使用Rust构建Python原生库?注意,不是动态链接库!!!

参考文档&#xff1a;https://github.com/PyO3/pyo3 创建python虚拟环境&#xff1a; conda create --name pyo3 python3.11.7激活虚拟环境&#xff1a; conda activate pyo3安装依赖&#xff1a; pip install maturin初始化项目&#xff1a; maturin init构建项目&#x…