C语言数据结构基础————二叉树学习笔记(四)简单的OJ题目练习

1.单值二叉树

965. 单值二叉树 - 力扣(LeetCode)

建立一个新的函数,用函数传参的方法来记录val的值

如上一篇最后的对称二叉树的习题,建立新的函数来传参

多采用使用反对值的方法,因为如果是相等return true的话,没有实质性的作用 

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

2.前序遍历

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

力扣的统一要求,凡是返回数组,一定要返回数组大小

这样还能真的返回returnSize的大小吗? 

经典的错误,标准的零分:传值调用!!

力扣是希望我们在函数内部修改好returnsize的值,这样他才能查看数组的大小以便于访问。

那我们怎么确定这个值大小呢?换句话问,我们如何开辟这个空间呢? 

当然是不建议一次性开一个很大的数组来保证足够使用的。

                   

我们可以模仿顺序表扩容的功能进行realloc,或者直接写一个计数节点个数的函数(此功能在上一篇中有讲)。

                                                                问题出在哪? 

i成为一个局部变量,每次的值都不会改变。

我们任然需要采用传址调用的方法:

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

 void _preorderPut(struct TreeNode* root,int* arr,int* pi){
    //前序遍历的顺序是根,左子树,右子树
    if(root==NULL){
        return;
    }
    arr[(*pi)++]=root->val;
    _preorderPut(root->left,arr,pi);
    _preorderPut(root->right,arr,pi);
 }

int* preorderTraversal(struct TreeNode* root, int* returnSize) {
    *returnSize=TreeNodeSize(root);
    int* arr=(int*)malloc(sizeof(int)*(*returnSize));
    //只创建一次数组即可,所以真正的遍历还是应当使用子函数
    int i=0;
    int* pi=&i;
     _preorderPut(root,arr,pi);
     return arr;
}

3.判断是否为子树

572. 另一棵树的子树 - 力扣(LeetCode)

为空、树的比较是最小子问题,isSubtree(left or right)是递归子问题。

我们应该把这两个问题分开,不要将树的比较嵌套进递归中,而应该分隔开两个逻辑。此处树的比较非常类似于前面题目的:

                                                    root1->val==root2->val

只不过是将数值的比较换做了整个子树的比较,我们直接复用之前写好的比较树的函数即可

利用之前的函数:判断树是否相同。遍历主树,将主树的每个值与subroot相比较。

                                                         

一如既往,在二叉树中空一直都是最小子问题,但是此处的空该return false还是return true呢?

根据题目描述,root不可能为空 

满足一次return true就会一直在

                       

每一次函数调用的“isSubtree”语句上做返回,层层返回,直到返回到函数外部

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

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

4.二叉树的创建和销毁(非递归)

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

这个题完成创建

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

typedef struct BTNode {
    struct BTNode* left;
    struct BTNode* right;
    char val;
}BTNode;

BTNode* CreatTree(char* arr, int* pi) {
    if (arr[*pi] == '#') {
        (*pi)++;
        return NULL;
    }
    BTNode* newnode = (BTNode*)malloc(sizeof(BTNode));
    newnode->val = arr[(*pi)++];
    newnode->left = CreatTree(arr, pi);
    newnode->right = CreatTree(arr, pi);
    return newnode;
}

void Inorder(BTNode* tree, char* arr) {//中序是左子树 根 右子树
    if (tree == NULL) {

        return;
    }
    Inorder(tree->left, arr);
    printf("%c ", tree->val);
    Inorder(tree->right, arr);
}

int main() {
    char arr[100];
    scanf("%s", arr);
    int i = 0;
    BTNode* tree = CreatTree(arr, &i);
    Inorder(tree, arr);

    return 0;
}

二叉树的销毁:

前序也能销毁,但是很麻烦,需要变量记录指针。

后序更佳:

后序+队列(利用队列的先进先出)

采取出来一个,带入自己的左右子节点 

注意声明和定义的分离

将树节点指针当作队列节点的值放入队列中。

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

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

相关文章

芯片公司SAP管理架构:科技与管理的完美融合

在当今日新月异的科技时代&#xff0c;芯片公司作为信息技术领域的核心力量&#xff0c;其运营管理的复杂性日益凸显。SAP管理架构作为一种高效的企业资源规划系统&#xff0c;为芯片公司提供了强大的管理支持。本文将为您科普芯片公司SAP管理架构的相关知识。 SAP管理架构是一…

leetcode每日一题1969

目录 一.题目原型&#xff1a; 二思路解析&#xff1a; 三.代码实现: 一.题目原型&#xff1a; 二思路解析&#xff1a; 灵神的做法非常让人惊叹&#xff1a; 理解就是&#xff0c;如果一个数大于另一个数要交换的1的权重&#xff0c;那么他们的乘积就变小。 那么一个大的数…

React 中的懒加载(Lazy Load)

React 中的懒加载&#xff08;Lazy Load&#xff09; 在大型的 React 应用中&#xff0c;为了提高页面加载速度和性能&#xff0c;我们经常会使用懒加载技术来延迟加载组件或资源。懒加载可以将页面初始加载的内容减少到最小&#xff0c;只在需要时再动态加载额外的组件或资源…

js实现hash路由原理

一、简单的上下布局&#xff0c;点击左侧导航&#xff0c;中间内容跟对变化&#xff0c;主要技术使用js检测路由的onhashchange事件 效果图 二、话不多说&#xff0c;直接上代码 <!DOCTYPE html> <html lang"zh"><head><meta charset"…

Wireshark 抓包工具与长ping工具pinginfoview使用,安装包

一、Wireshark使用 打开软件&#xff0c;选择以太网 1、时间设置时间显示格式 这个时间戳不易直观&#xff0c;我们修改 2、抓包使用的命令 1&#xff09;IP地址过滤 ip.addr192.168.1.114 //筛选出源IP或者目的IP地址是192.168.1.114的全部数据包。 ip.sr…

Win10中IIS服务如何部署c#服务

1、将项目打包发布 注意发布位置 2、打开搜索搜索计算机管理 3、点击服务和应用程序 4、点击internet information service 5、点击网站再点击添加网站 6、添加网站名称:opm 添加网站路径(即刚才发布路径) 输入ip地址:自己电脑ip 配置端口号5052 最后点击确认 7、…

vue3中如何实现多个侦听器(watch)

<body> <div id"app"><input type"button" value"更改名字" click"change"> </div> <script src"vue.js"></script> <script>new Vue({el: #app,data: {food: {id: 1,name: 冰激…

类型双关联合体(C++基础)

类型双关 类型双关就是在同样的一个内存中&#xff0c;存储的数据可以用做不同类型的表述。用在底层性能优化的时候&#xff0c;一般使用的时候要非常小心&#xff0c;因为不当的使用可能导致程序的不稳定和不可预测的行为。 int a 5;//double value (double)a;double value…

卷王的自述,我为什么这么卷?

&#x1f345; 视频学习&#xff1a;文末有免费的配套视频可观看 &#x1f345; 关注公众号【互联网杂货铺】&#xff0c;回复 1 &#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 前段时间去面试了一个公司&#xff0c;成功拿到了offer&#xf…

2024年3月22蚂蚁新村今日答案:以下哪一项是陕西省的非遗美食?

2024年3月22日蚂蚁新村今日问题的正确答案如下&#xff1a; 问题&#xff1a;以下哪一项是陕西省的非遗美食&#xff1f; 选项&#xff1a;驴肉火烧 水盆羊肉 答案&#xff1a;水盆羊肉 解析&#xff1a;水盆羊肉是陕西省的非遗美食。水盆羊肉是陕西省的一道传统著名饭食&a…

2024年Jira全面解析:从 Jira 的概念到优缺点、最新政策

Jira是澳大利亚的Atlassian公司开发的一款项目管理软件&#xff0c;名字来源于日文中“哥斯拉”的称呼“Gojira”。Jira不仅可以追踪缺陷和问题&#xff0c;还能管理项目。很多企业还将JIRA用于一些特殊的场景&#xff0c;比如作为仓库自动化工具、管理文档流程、优化费用等等。…

NCCL 简介

文章目录 前言1. NCCL简介2. 如何使用NCCL 前言 NCCL 源码解析总目录 简单写几个重点。 如有问题&#xff0c;请留言指正。 1. NCCL简介 NCCL主要用来集体通信的&#xff0c;提高多个CUDA设备的通信效率 发展&#xff0c;2.X 主要支持多个节点 NCCL中使用的通信总线…

最快的 Python API 框架之一:简单、现代、高性能 | 开源日报 No.207

tiangolo/fastapi Stars: 68.1k License: MIT fastapi 是一个现代、高性能、易学习、快速编码且适用于生产环境的框架。 其主要功能和核心优势包括&#xff1a; 高性能&#xff1a;与 NodeJS 和 Go 相当&#xff0c;是最快的 Python 框架之一。编码速度快&#xff1a;开发特性…

【CVTE 一面凉经Ⅰ】循环依赖如何解决

目录 一.&#x1f981; 开始前的废话二. &#x1f981; 什么是循环依赖&#xff1f;三. &#x1f981;Spring 容器解决循环依赖的原理是什么?五. &#x1f981; 三级缓存解决循环依赖的原理六. &#x1f981; 由有参构造方法注入属性的循环依赖如何解决&#xff1f;七.&#x…

微前端架构

介绍 微前端的概念是由ThoughtWorks在2016年提出的&#xff0c;它借鉴了微服务的架构理念&#xff0c;核心在于将一个庞大的前端应用拆分成多个独立灵活的小型应用&#xff0c;每个应用都可以独立开发、独立运行、独立部署&#xff0c;再将这些小型应用融合为一个完整的应用&am…

当内外网的域名相同时,如何在外网解析同域名的网址

当内部网络和外部网络存在相同的域名&#xff0c;并且希望内部用户通过内部DNS服务器解析到外部网络上的该域名对应的公网IP地址时&#xff0c;需要在内部DNS服务器上采取一些特殊配置策略来实现这一目标。以下是一种通用的解决方案&#xff1a; 条件转发&#xff08;Condition…

鸿蒙一次开发,多端部署(四)工程管理

DevEco Studio的基本使用&#xff0c;请参考DevEco Studio使用指南。本章主要介绍如何使用DevEco Studio进行多设备应用开发。 说明&#xff1a; 本章的内容基于DevEco Studio 3.1.1 Release版本进行介绍&#xff0c;如您使用DevEco Studio其它版本&#xff0c;可能存在文档与产…

COMPOSITE SLICE TRANSFORMER

Composite Slice Attention (CSA) 辅助信息 作者未提供代码

第十四届蓝桥杯JavaB组省赛真题 - 蜗牛

dp[i][0] 状态转移方程&#xff1a; 1. 从上一个竹竿的底部转移过来&#xff0c;即&#xff1a; dp[i][0]dp[i−1][0]x[i]−x[i−1]; 2. 从上一个竹竿的传送门转移过来&#xff0c;即&#xff1a; dp[i][0]dp[i−1][1]b[i]/1.3; dp[i][1] 状态转移方程&#xff1a; 1. 从上一…

Java集合框架-读书笔记

Java集合框架 数据结构是以某种形式将数据组织在一起的集合。数据结构不仅可以存储数据 并且可以对数据进行访问和处理操作。 eg&#xff1a;Arraylist是将数据保存在线性表的数据结构 其实java中还提供了一些第一数据进行操作和存储的数据结构 这些数据结构被称为 java集合…