【数据结构】——二叉树OJ题

news2024/12/27 16:01:04

文章目录

    • 前言
    • 1. 单值二叉树
    • 2. 检查两颗树是否相同
    • 3. 判断一棵树是否为另一颗树的子树
    • 4. 对称二叉树
    • 5. 平衡二叉树
    • 6. 二叉树的前序遍历
    • 7. 二叉树的中序遍历
    • 8. 二叉树的后序遍历
    • 9. 二叉树的构建及遍历

前言

我们先想想二叉树我们学习了哪些内容再来做题哈

在这里插入图片描述

其实学习二叉树重要的思想——递归,我们可以通过后面的题目慢慢的理解

1. 单值二叉树

OJ链接:965. 单值二叉树 - 力扣(LeetCode)

好的,我们一起来看一下题目,题目是这样说的

题目描述

如果二叉树每个节点都具有相同的值,那么该二叉树就是单值二叉树,只有给定的树是单值二叉树时,才返回 true;否则返回 false

示例

在这里插入图片描述

思路分析

我们想一下,单值二叉树是不是一棵树的所有节点的值都相同,当且仅当对于树上的每一个子树的两个孩子,它们也都有相同的值

因此,我们可以对树进行一次深度优先搜索,当搜索到节点root时,我们检查root的左孩子和右孩子是否相同,不相同则返回false,直到检查了所有的节点,所以我们就可以进行递归遍历,每次比较根节点和左右孩子的val值是否相等,不相等就返回false,然后递归比较左子树和右子树

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

我们现在来尝试画一下递归展开图,更方便我们理解

[3,3,3,null,null,2,3] 举例

在这里插入图片描述

2. 检查两颗树是否相同

OJ链接:100. 相同的树 - 力扣(LeetCode)

好的,我们一起来看一下题目,题目是这样说的

题目描述

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

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

示例

在这里插入图片描述

思路分析

我们先可以想象一下,如果两棵树相同的话,应该会有哪些情况

如果两个树都是空树的话,就是相同的,那如果一棵树是空树,另一颗不是空树,那肯定不是相同的

如果两个二叉树都不为空,那么首先判断它们的根节点的值是否相同,若不相同则两个二叉树一定不同,若相同,再分别判断两个二叉树的左子树是否相同以及右子树是否相同。这是一个递归的过程,因此可以使用深度优先搜索,递归地判断两个二叉树是否相同

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

这个递归展开和上一个题一样,我们就不画了,留给大家画一下哈

3. 判断一棵树是否为另一颗树的子树

OJ链接:572. 另一棵树的子树 - 力扣(LeetCode)

好的,我们一起来看一下题目,题目是这样说的

题目描述

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

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

示例

在这里插入图片描述

思路分析

由于root和subRoot中可能含有一个和多个值相同的节点,所以判断不相等的时候,又要返回原来的根节点,所以我们可以这道题利用上一题的代码,我们的思路为不断的比较root这棵树以每一个节点作为根节点,判断是否和subRoot相等,相等就返回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);
}

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

4. 对称二叉树

OJ链接:110. 平衡二叉树 - 力扣(LeetCode)

好的,我们一起来看一下题目,题目是这样说的

题目描述

给你一个二叉树的根节点 root , 检查它是否轴对称

示例

在这里插入图片描述

思路分析

这道题和判断两棵树是否相等的思路一致,只是有一些细节有所不同。

对称二叉树是最左边和最右边的节点相同,所以我们就可以拿第一棵树的左子树和第二棵树的右子树进行比较,拿第一棵树的右子树和第二棵树的左子树进行比较,不相同就返回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->right) &&
    	isSameTree(p->right,q->left);
}

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

5. 平衡二叉树

OJ链接:110. 平衡二叉树 - 力扣(LeetCode)

好的,我们一起来看一下题目,题目是这样说的

题目描述

给定一个二叉树,判断它是否是 平衡二叉树

平衡二叉树 是指该树所有节点的左右子树的深度相差不超过 1。

示例

在这里插入图片描述

思路分析

这道题中的平衡二叉树的定义是:二叉树的每个节点的左右子树的高度差的绝对值不超过 1,则二叉树是平衡二叉树。根据定义,一棵二叉树是平衡二叉树,当且仅当其所有子树也都是平衡二叉树,因此可以使用递归的方式判断二叉树是不是平衡二叉树

对于当前遍历到的节点,首先计算左右子树的高度,如果左右子树的高度差是否不超过 1,再分别递归地遍历左右子节点,并判断左子树和右子树是否平衡。这是一个自上而下的递归的过程

int Rootheight(struct TreeNode* root){
    if(root == NULL)
        return 0;
    else
        return fmax(Rootheight(root->left),Rootheight(root->right)) + 1;
}

bool isBalanced(struct TreeNode* root) {
    if(root == NULL)
        return true;
    else
        return fabs(Rootheight(root->left) - Rootheight(root->right)) <=1
    && isBalanced(root->left) && isBalanced(root->right);
}

6. 二叉树的前序遍历

OJ链接:144. 二叉树的前序遍历 - 力扣(LeetCode)

好的,我们一起来看一下题目,题目是这样说的

题目描述

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

示例

在这里插入图片描述

思路分析

二叉树的前序遍历我们已经非常熟悉,这里需要注意:

  1. 由于二叉树的节点数是未知的,为了不浪费空间,我们可以先求出二叉树的节点数,然后开辟对应大小的空间
  2. 由于数据存储在一个数组中,所以我们需要一个变量i来控制数组的下标,由于在递归调用的过程中对形参的改变不会改变影响实参,所以这里我们需要传递i的地址
int TreeSize(struct TreeNode* root)
{
    if(root == NULL)
        return 0;
    else
        return TreeSize(root->left) + TreeSize(root->right) + 1;
}

void preorder(struct TreeNode* root, int* a, int* pi)
{
    if(root == NULL)
        return;
    a[*pi] = root->val;
    (*pi)++;
    preorder(root->left,a,pi);
    preorder(root->right,a,pi);
}

int* preorderTraversal(struct TreeNode* root, int* returnSize) {
    int size = TreeSize(root);
    int* a = (int*)malloc(sizeof(int) * size);
    int i = 0;
    preorder(root,a,&i);
    *returnSize = size;
    return a;
}

那让我们看看前序遍历的递归展开图

在这里插入图片描述

7. 二叉树的中序遍历

OJ链接:94. 二叉树的中序遍历 - 力扣(LeetCode)

好的,我们一起来看一下题目,题目是这样说的

题目描述

给定一个二叉树的根节点 root ,返回它的 中序 遍历 。

示例

在这里插入图片描述

二叉树的中序遍历和前序遍历一样,只是访问节点的顺序不同

代码实现如下

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

void preorder(struct TreeNode* root, int* a, int* pi)
{
    if(root == NULL)
        return;
    preorder(root->left,a,pi);
    a[*pi] = root->val;
    (*pi)++;
    preorder(root->right,a,pi);
}

int* inorderTraversal(struct TreeNode* root, int* returnSize) {
    int size = TreeSize(root);
    int* a = (int*)malloc(sizeof(int) * size);
    int i = 0;
    preorder(root,a,&i);
    *returnSize = size;
    return a;
}

8. 二叉树的后序遍历

OJ链接:145. 二叉树的后序遍历 - 力扣(LeetCode)

好的,我们一起来看一下题目,题目是这样说的

题目描述

给你一棵二叉树的根节点 root ,返回其节点值的 后序遍历

示例

在这里插入图片描述

二叉树的中序遍历和前序遍历一样,只是访问节点的顺序不同

代码实现如下

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

void preorder(struct TreeNode* root, int* a, int* pi)
{
    if(root == NULL)
        return;
    preorder(root->left,a,pi);
    preorder(root->right,a,pi);
    a[*pi] = root->val;
    (*pi)++;
}

int* postorderTraversal(struct TreeNode* root, int* returnSize) {
    int size = TreeSize(root);
    int* a = (int*)malloc(sizeof(int) * size);
    int i = 0;
    preorder(root,a,&i);
    *returnSize = size;
    return a;
}

9. 二叉树的构建及遍历

OJ链接:二叉树遍历_牛客题霸_牛客网 (nowcoder.com)

好的,我们一起来看一下题目,题目是这样说的

题目描述

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

输入描述:

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

输出描述:

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

在这里插入图片描述

思路分析

这道题目是前序建立二叉树和中序遍历,我们写成两个子函数即可,对于二叉树的创建,字符为‘#’说明节点为空,我们直接返回即可,然后依次递归创建节点即可

代码实现如下

#include <stdio.h>
#include <stdlib.h>

typedef char BTDataType;
typedef struct BinaryTreeNode
{
    BTDataType data;
    struct BinaryTreeNode* left;
    struct BinaryTreeNode* right;
} BTNode;

// 构建二叉树
BTNode* BTreeCreate(char* a, int* pi)
{
    if (a[*pi] == '#')
    {
        (*pi)++;
        return NULL;
    }
    // 创建根节点
    BTNode* root = (BTNode*)malloc(sizeof(BTNode));
    if (root == NULL) 
    {
        perror("malloc fail");
        exit(-1);
    }
    root->data = a[*pi];
    (*pi)++;
    // 创建左子树和右子树
    root->left = BTreeCreate(a, pi);
    root->right = BTreeCreate(a, pi);
    return root;
}
// 二叉树中序遍历
void InOrder(BTNode* root)
{
    if (root == NULL)
    {
        return;
    }
    // 先访问左子树,再访问根节点,最后访问右子树
    InOrder(root->left);
    printf("%c ", root->data);
    InOrder(root->right);
}

int main()
{
    char str[100];
    scanf("%s", str);
    // 创建二叉树
    int i = 0;
    BTNode* root = BTreeCreate(str, &i);
    // 二叉树的中序遍历
    InOrder(root);

    return 0;
}

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

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

相关文章

JavaScript (十四)——JavaScript typeof和类型转换

目录 JavaScript typeof, null, 和 undefined typeof 操作符 null undefined undefined 和 null 的区别 JavaScript 类型转换 JavaScript 数据类型 JavaScript 类型转换 将数字转换为字符串 将布尔值转换为字符串 将日期转换为字符串 将字符串转换为数字 一元运算符…

新浪微博笔试秋招管培笔试测评肯耐珂萨题型题库解析

新浪微博的笔试是该公司用于筛选潜在候选人的重要环节&#xff0c;主要目的在于评估应聘者的逻辑推理、图表计算和文字理解能力。以下是新浪微博笔试题型的详细解析&#xff1a; 1. 逻辑推理 - **题型概述**&#xff1a;逻辑推理部分通常包含20题&#xff0c;考察应试者的逻…

算法板子:树形DP、树的DFS——树的重心

思想&#xff1a; 代码&#xff1a; #include <iostream> #include <cstring> using namespace std;const int N 1e5 10;// vis标记当前节点是否被访问过; vis[1]true代表编号为1的节点被访问过 bool vis[N]; // h数组为邻接表; h数组上的每个坑位都串了一个单链…

商业购物中心开业活动方案怎么写?附230个案例

商业购物中心开业活动方案的撰写是一个综合性的过程&#xff0c;需要结合购物中心的特点、目标消费群体、市场环境以及活动目的等多方面因素进行考虑。 以下是一个详细的撰写指南&#xff0c;带你一步步了解如何撰写一个成功的商业购物中心开业活动方案。 码字不易&#xff0…

处理kkFileView的com.aspose:aspose-cad:iar:23.9 not found 问题

背景: 一款很强大的开源的文件预览工具包下载地址:aspose-cad-23.9.jar 启动成功界面: 一、.问题描述:com.aspose:aspose-cad:iar:23.9 not found 处理方案:直接下载aspose-cad-23.9.jar 安装到本地 二、处理步骤(win环境):1.安装maven 下载地址:https://maven.apach…

CSP-J 复赛 模拟题

1.生产计划&#xff1a; 样例 #1 样例输入 #1 2 4 5 6 12 1 3 6 15 8 1 3 100 3 200 4 300 6 100 样例输出 #1 YES NO 2.分组和为3&#xff1a; 样 例 # 1 样 例 输 入 # 1 5 1 1 1 2 1 样 例 输 出 # 1 2 样 例 # 2 样 例 输 入 # 2 7 2 2 1 1 2 1 1 样 例 输 出 # …

2024最简七步完成 将本地项目提交到github仓库方法

2024最简七步完成 将本地项目提交到github仓库方法 文章目录 2024最简七步完成 将本地项目提交到github仓库方法一、前言二、具体步骤1、github仓库创建2、将远程仓库拉取并合并&#xff08;1&#xff09;初始化本地仓库&#xff08;2&#xff09;本地仓库与Github仓库关联&…

Linux驱动开发—并发与竞争,原子操作,自旋锁,信号量详解

1.并发与并行的概念 并发是指在同一时间段内&#xff0c;多个任务交替执行。并发可以发生在单核处理器上&#xff0c;通过任务切换实现 并行是指在同一时间段内&#xff0c;多个任务同时执行。并行可以发生在多核处理器上&#xff0c;例如下图任务1 和任务3同时进行&#xff0…

JAVA基础知识点3 (String 和 StringBuffer 以及 StringBuilder 的特点以及区别)

1&#xff0c;String 和 StringBuffer 以及 StringBuilder 的特点 &#xff08;1&#xff09;String的特点&#xff1a;String是final修饰的字符序列是不可改变的&#xff0c; 是字符串常量&#xff0c;一旦初始化就不可以被更改,因此是线程安全的 因为是常量每次对其操作都会…

C++必修:STL之vector的模拟实现

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 所属专栏&#xff1a;C学习 贝蒂的主页&#xff1a;Betty’s blog 为了让我们更加深入理解vector&#xff0c;接下来我们将模拟实现一个简易版的vect…

龙迅#LT8918适用于TTL/LVDS转MIPIDSI/CSI应用方案,分辨率高达1080P@60HZ,可提供技术支持!

1. 描述 Lontium LT8918 是一款高性能 MIPIDSI/CSI-2 发射器&#xff0c;适用于移动显示面板或相机应用。 LT8918 的 TTL 输入在 SDR 或 DDR 采样下支持 24 位 RGB 和 BT656/1120 视频格式。最大输入像素时钟频率为 SDR 148.5MHz 或 DDR 74.25MHz&#xff0c;适用于1080P60Hz高…

PCL从理解到应用【08】 点云特征 | 法线估计 | 主曲率估计

前言 在PCL中&#xff0c;有多种方法和函数可以用来提取点云特征&#xff0c;本文介绍几何特征。 其中&#xff0c;几何特征主要包括法线估计和主曲率估计。 这些特征能够描述点云表面的几何形状&#xff0c;常用于进一步的点云处理和分析&#xff0c;如配准、分割和物体识别…

为什么 Kubernetes 是现代开发的必备工具

引言 在现代软件开发中&#xff0c;容器已经成为打包和运行应用程序的标准方式。然而&#xff0c;在生产环境中&#xff0c;管理这些运行中的容器并确保服务的高可用性和稳定性并不是一件容易的事。比如&#xff0c;当一个容器发生故障时&#xff0c;需要快速启动另一个容器来代…

C++ std::atomic和std::mutex

C11 引入了两个重要的同步机制用于多线程编程&#xff1a;std::atomic 和 std::mutex。它们各自适用于不同的并发控制需求&#xff0c;并在实现和使用上有很大的不同。 1. 目的和用途 std::atomic: 设计目的&#xff1a;为原子操作提供支持&#xff0c;保证对变量的操作&#…

Python试讲

Python试讲 导语Python简介Python及其特点如何使用Python Python与计算计算变量 导语 本次试讲内容如下&#xff1a;Python简介与使用&#xff0c;Python与基本运算 辅助教材为 《趣学Python编程》和《Python编程从入门到实践》 Python简介 Python是目前入门最简单最好学的…

FVM安装及配置

一、下载fvm 包 git&#xff1a;Release fvm 3.1.7 leoafarias/fvm GitHub 解压到本地文件夹&#xff0c;然后添加环境变量 管理员模式打开cmd&#xff0c;查看是否成功 fvm --version 二、安装Dart SDK 下载Dart SDK&#xff1a;Dart for Windows 三、安装GIT 四、指定…

python3.10安装geopandans实战笔记

1.geopandans安装所需软件库版本 python3.10 GDAL-3.4.3-cp310-cp310-win_amd64.whl【手动下载】 Fiona-1.8.21-cp310-cp310-win_amd64.whl【手动下载】 shapely-2.0.2-cp310-cp310-win_amd64.whl【手动下载】 pyproj 手动下载地址&#xff1a;https://download.csdn.net/down…

range和enumerate的区别

range通过索引遍历元素&#xff0c;属于间接访问。 enumerate直接遍历元素&#xff0c;效率稍高&#xff0c;代码简洁。range输出的是元素的索引。 enumerate输出的是元素的索引和元素。 参考&#xff1a;range与enumerate的区别_enumerate和range-CSDN博客

(一)springboot2.7.6集成activit5.23.0之集成引擎

集成引擎很简单。 首先是创建springboot项目然后引入相关依赖就完成了。pom.xml如下&#xff1a; <project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation"http://maven.…

鸿蒙应用框架开发【媒体库视频】 UI框架

媒体库视频 介绍 本示例使用Video组件展示了视频组件的基本功能&#xff0c;包括视频组件化&#xff0c;全屏化&#xff0c;窗口化&#xff0c;上下轮播视频等。 效果预览 使用说明&#xff1a; 进入首页点击播放按键&#xff1b;点击视频播放按钮&#xff0c;视频开始播放…