简单介绍二叉树

news2025/1/11 9:09:55

前言

学习数据结构,二叉树是一大难点,也是一大重点,小伙伴们和我一起看看二叉树的知识吧!

本文代码是Java。


目录

前言

一、什么是二叉树

二、二叉树的遍历

(一)前序遍历

(二)中序遍历

(三)后序遍历

(四)层序遍历

(五)依据前序遍历和中序遍历求二叉树

(六)依据中序遍历和后序遍历求二叉树

结语


一、什么是二叉树

二叉树是指只有一个根节点并且只有两个孩子节点的树。

例如:

二叉树中有两种特殊的二叉树,分别是满二叉树和完全二叉树。

  1. 满二叉树

一个二叉树,如果每一个层的结点数都达到最大值,则这个二叉树就是满二叉树。

        2.完全 二叉树

一棵深度为k的有n个结点的二叉树,对树中的结点按从上至下、从左到右的顺序进行编号,如果编号为i(1≤i≤n)的结点与满二叉树中编号为i的结点在二叉树中的位置相同,则这棵二叉树称为完全二叉树。

注意事项:

  1. 一棵n个结点的树有n-1条边;
  2. 满二叉树的层数为k,则结点总数为(2^k)-1;
  3. 若根节点的层数为1,则非空二叉树的第i层最多(2^(i-1))-1个结点;
  4. 叶子节点n0,度为2的结点n2,则n0=n2+1,任何二叉树都满足这个特性;
  5. 具有n个结点的完全二叉树的深度k为log2(n+1)向上取整;
  6. 对于完全二叉树,
  •                              当根节点为0时,已知父节点为i,则左孩子节点为2i+1,右孩子节点为2i+2;
  •                              当根节点为1时,已知父节点为i,则左孩子节点为2i,右孩子节点为2i+1;

二、二叉树的遍历

以该二叉树为例子:

(一)前序遍历

前序遍历始终遵守 根节点 -> 左节点 -> 右节点的规律;

例如此二叉树通过前序遍历得出的顺序是:ADBCF.

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

递归版本:

class Solution {
    public List<Integer> preorderTraversal(TreeNode root) {
        func(root);
        return list;
    }

    List<Integer> list = new ArrayList<>();
    public void func(TreeNode root){
        if(root == null){
            return;
        }
        list.add(root.val);
        func(root.left);
        func(root.right);
    }
}

非递归版本:

class Solution {
    public List<Integer> preorderTraversal(TreeNode root) {
        List<Integer> list = new ArrayList<>();
        Stack<TreeNode> stack = new Stack<>();
        TreeNode tmp = null;
        while(!stack.empty() || root!=null){           
            while(root != null){
                stack.push(root);
                list.add(root.val);
                root = root.left;
            }
            root = stack.pop();
            root = root.right;
        }
        return list;
    }
}

(二)中序遍历

中序遍历始终遵守 左节点 -> 根节点 -> 右节点的规律;

例如此二叉树通过中序遍历得出的顺序是:DBAFC.

94. 二叉树的中序遍历 - 力扣(Leetcode)

递归版本:

class Solution {
    public List<Integer> inorderTraversal(TreeNode root) {
        func(root);
        return list;
    }
    
    List<Integer> list = new ArrayList<>();
    public void func(TreeNode root){
        if(root == null){
            return;
        }
        func(root.left);
        list.add(root.val);
        func(root.right);
    }
}

非递归版本:

class Solution {
    public List<Integer> inorderTraversal(TreeNode root) {
         List<Integer> list = new ArrayList<>();
        Stack<TreeNode> stack = new Stack<>();
        while(!stack.empty() || root!=null) {
            while(root != null){
                stack.push(root);
                root = root.left;
            }
            root = stack.pop();
            list.add(root.val);
            root = root.right;
        }
        return list;
    }
}

(三)后序遍历

后序遍历始终遵守 左节点 -> 右节点 -> 根节点 的规律;

例如此二叉树通过后序遍历得出的顺序是:BDFCA.

145. 二叉树的后序遍历 - 力扣(Leetcode)

递归版本:

class Solution {
    public List<Integer> postorderTraversal(TreeNode root) {
        func(root);
        return list;
    }
    List<Integer> list = new ArrayList<>();
    public void func(TreeNode root){
        if(root == null){
            return;
        }
        func(root.left);
        func(root.right);
        list.add(root.val);
    }
}

非递归版本:

class Solution {
    public List<Integer> postorderTraversal(TreeNode root) {
        List<Integer> list = new ArrayList<>();
        Stack<TreeNode> stack = new Stack<>();
        TreeNode tmp = null;
        while(!stack.empty() || root != null){
            while(root != null){
                stack.push(root);
                root = root.left;
            }
            root = stack.pop();
            if (root.right == null || root.right == tmp) {
                list.add(root.val);
                tmp = root;
                root = null;
            } else {
                stack.push(root);
                root = root.right;
            }
        }

        return list;
    }
}

(四)层序遍历

层序遍历始终遵守 从根节点开始遍历,一层一层往下遍历 的规律;

例如此二叉树通过层序遍历得出的顺序是:ADCBF.

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

代码:

class Solution {
    public List<List<Integer>> levelOrder(TreeNode root) {
        Queue<TreeNode> queue = new LinkedList<>();
        List<List<Integer>> list = new LinkedList<>();
        if(root != null){
            queue.offer(root);
        }
        while(!queue.isEmpty()){ 
            List<Integer> list1 = new LinkedList<>();
            int size1 = queue.size();
            while(size1-- > 0){
                root = queue.poll();
                list1.add(root.val);
                if(root.left != null){
                    queue.offer(root.left);
                }
                if(root.right != null){
                    queue.offer(root.right);
                }
            }        
            list.add(list1);    
        }
        return list;
    }
}

(五)依据前序遍历和中序遍历求二叉树

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

class Solution {
    public TreeNode buildTree(int[] preorder, int[] inorder) {
        if(inorder == null){
            return null;
        }
        return buildTreeChild(preorder,inorder, 0, inorder.length-1);
    }

    int pre = 0;
    public TreeNode buildTreeChild(int[] preorder, int[] inorder, int form, int end){
        if(form > end) return null;
        //if(pre >= preorder.length) return null;
        TreeNode tmp = new TreeNode(preorder[pre]);

        int i = findTree(inorder, preorder[pre],form,end);
        pre++;
        tmp.left = buildTreeChild(preorder,inorder, form, i-1);
        tmp.right = buildTreeChild(preorder,inorder, i+1, end);

        return tmp;
    }

    public int findTree(int[] inorder,int k, int index, int end) {
        for(int i = index; i<= end; i++){
            if(inorder[i] == k){
                return i;
            }
        }
        return -1;
    }
}

(六)依据中序遍历和后序遍历求二叉树

106. 从中序与后序遍历序列构造二叉树 - 力扣(Leetcode)

class Solution {
    public TreeNode buildTree(int[] inorder, int[] postorder) {
        if(inorder.length == 1) 
            return new TreeNode(inorder[0]);
        return buildTreeChild(inorder, postorder, 0, inorder.length-1, postorder.length-1);
    }

    int i = 0;
    public TreeNode buildTreeChild(int[] inorder, int[] postorder, int form, int end, int len){
        if(form > end) return null;
        if(i > len) return null;
        TreeNode tmp = new TreeNode(postorder[len-i]);

        int index = findTree(inorder, postorder[len-i],form,end);
        i++;
        tmp.right = buildTreeChild(inorder, postorder, index+1, end, len);
        tmp.left = buildTreeChild(inorder, postorder, form, index-1, len);

        return tmp;
    }

    public int findTree(int[] inorder, int key, int form, int end){
        for(int i = form; i <= end; i++){
            if(inorder[i] == key)
                return i;
        }
        return -1;
    }

}

结语

关于二叉树的知识点还有很多,例如红黑树、AVL树……

我们下期再见!

这篇博客如果对你有帮助,给博主一个免费的点赞以示鼓励,欢迎各位🔎点赞👍评论收藏⭐,谢谢!!!

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

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

相关文章

C++ 入门导引(这是一篇由GPT4写的文章)

C 应用场景 C 是一种广泛应用的编程语言&#xff0c;拥有多种使用场景。以下是 C 的一些主要应用场景&#xff1a; ​1. 游戏开发&#xff1a;C 常用于游戏开发&#xff0c;尤其是大型 3D 游戏。它可以轻松地与图形 API&#xff08;如 OpenGL 和 DirectX&#xff09;集成&…

element-ui树形控件el-tree详解

概述 这里我利用element-ui开发一个vue的树形组件 引入element-ui 安装element-plus cnpm install element-plus --save 安装按需导入 cnpm install -D unplugin-vue-components unplugin-auto-import 修改vite.config.js配置按需加载 import AutoImport from unplugin-a…

云原生架构:创新未来的应用开发和部署范式

点击上方“程序猿技术大咖”&#xff0c;关注并选择“设为星标” 回复“加群”获取入群讨论资格&#xff01; 摘要&#xff1a;本文深入探讨云原生架构的优势、实施指南以及关键技术和工具。通过容器化、微服务架构、持续交付和自动化管理等实践&#xff0c;云原生架构为企业提…

盘点一个Python列表的基础题目

点击上方“Python爬虫与数据挖掘”&#xff0c;进行关注 回复“书籍”即可获赠Python从入门到进阶共10本电子书 今 日 鸡 汤 随意春芳歇&#xff0c;王孙自可留。 大家好&#xff0c;我是皮皮。 一、前言 前几天在Python最强王者群【eric】问了一个Python列表基础的问题&#x…

web练习第二周

前言&#xff1a;&#xff08;博主个人学习笔记&#xff0c;不用看&#xff09;web练习第二周&#xff0c;仅做出前3题。相比于第一周&#xff0c;难度大幅增加&#xff0c;写题时就算看了wp还是像个无头苍蝇一样到处乱创&#xff0c;大多都是陌生知识点&#xff0c;工具的使用…

购买两块巧克力-第105场力扣夜喵双周赛-java双百方案

一、题目描述 给你一个整数数组 prices &#xff0c;它表示一个商店里若干巧克力的价格。同时给你一个整数 money &#xff0c;表示你一开始拥有的钱数。 你必须购买 恰好 两块巧克力&#xff0c;而且剩余的钱数必须是 非负数 。同时你想最小化购买两块巧克力的总花费。 请你…

【原创】浅谈EtherCAT主站EOE(上)-EOE网络

版权声明&#xff1a;本文为本文为博主原创文章&#xff0c;未经同意&#xff0c;禁止转载。如有问题&#xff0c;欢迎指正。博客地址&#xff1a;https://www.cnblogs.com/wsg1100/ 文章目录 一、EoE二、EoE服务规范EtherCAT主站如何提供EoE服务&#xff1f; 三、EoE网络EOE网…

我用GPT搭建了一个虚拟女友!

Datawhale干货 作者&#xff1a;仲泰&#xff0c;Datawhale成员 1. 作者知乎&#xff1a;https://www.zhihu.com/people/yong-tan-39-67 2.我用GPT搭建了一个虚拟女友-哔哩哔哩&#xff1a;https://b23.tv/GYYwMcq 3. 五月学习&#xff1a;ChatGPT应用组队学习来了&#xff01…

原生canvas标签画线——直线、平行线(设置不同颜色和宽度)

1.一条直线 效果图如下&#xff1a; 代码如下&#xff1a; <!--* Author: your name* Date: 2023-05-24 17:50:28* LastEditTime: 2023-05-24 18:06:39* LastEditors: localhost* Description: In User Settings Edit* FilePath: /canvas/day01/体验canvas.html --> &l…

Python潮流周刊#3:PyPI 的安全问题

△点击上方“Python猫”关注 &#xff0c;回复“1”领取电子书 你好&#xff0c;我是猫哥。这里记录每周值得分享的 Python 及通用技术内容&#xff0c;部分为英文&#xff0c;已在小标题注明。&#xff08;标题取自其中一则分享&#xff0c;不代表全部内容都是该主题&#xff…

网站部署与上线(2)远程连接云服务器或虚拟机

文章目录 搭建服务器部署环境配置pm2 可能听说过Windows系统提供的远程桌面。实际上&#xff0c;Linux中也提供了类似的功能&#xff0c;其远程连接基于命令行。 在Windows端连接Linux需要使用SSH软件&#xff0c;最流行的有Xshell和SecureCRT。 首先确定需要连接的云服务器或虚…

Pyside6-第一篇-创建第一个窗口

Hi&#xff0c;今天起开始更新Pyside6教程了&#xff0c;从0-1开始更新&#xff0c;过程比较的久&#xff0c;一点点来。 今天&#xff0c;我们先来搭建环境。 我的环境&#xff1a; ❝ pycharm 2021.3.3(版本随意&#xff0c;只要不是很低就行)Python版本3.95Pyside版本6.50 ❞…

【FreeRTOS】——中断优先级设置中断相关寄存器临界段代码保护调度器挂起与恢复

目录 前言&#xff1a; 一、中断优先级设置 二、中断相关寄存器&#xff08;STM32-Cortex M3&#xff09; 三、临界段代码保护 四、任务调度器的挂起和恢复 总结&#xff1a; 前言&#xff1a; 博客笔记根据正点原子视频教程编辑&#xff0c;仅供学习交流使用&#xff0…

电子器件系列38:mos管散热片

板子上需要用到一个封装为to220的mos管&#xff0c;还得立起来散热&#xff0c;得要加一个散热片。 散热片简介&#xff0c;分类&#xff1f;用途&#xff1f;如何使用&#xff1f;封装&#xff1f;使用注意事项&#xff1f; 简介&#xff1a; mos散热片是一种给电器中的易发热…

线程池实现

一、线程池介绍 1&#xff09;应用场景 当并发数很多的时候&#xff0c;并且每个线程执行时间很短的任务&#xff0c;这样就会频繁创建线程&#xff0c;而这样的频繁创建和销毁线程会大大降低系统的执行效率。对于这种场景我们可以使用线程池来复用之前创建的线程&#xff0c…

Linux—基础篇:目录结构

1、基本介绍 1、linux的文件系统是采用级层式的树状目录结构&#xff0c;在此结构的最上层是根目录“/”,然后在此目录下创建其他目录 2、在Linux的世界里&#xff0c;一切皆文件&#xff01;&#xff01;&#xff01; 2、具体的目录结构 不用背&#xff0c;知道即可 2.1、…

价值1000元的稀有二开版的无限坐席在线客服系统源码+教程

demo软件园每日更新资源,请看到最后就能获取你想要的: 1.价值1000元的稀有二开版的无限坐席在线客服系统源码教程 价值1000元的稀有二开版的无限坐席在线客服系统源码 直接一键安装的&#xff0c;启动两个端口就行了&#xff0c;安装倒是简单 类型&#xff1a;在线客服系统 …

MathType7精简版数学公式编辑器

许多简单的数学公式&#xff0c;我们可以使用输入法一个个找到特殊符号并输入&#xff0c;但是对于高等数学中较多复杂的公式符号&#xff0c;是很难使用输入法完成的。那么&#xff0c;我们就需要借助公式编辑器&#xff0c;这里推荐一款我自己正在使用的MathType。 MathType是…

Redis单机数据库

文章目录 一、Redis数据库Redis数据库redisDb数据库键空间——dict过期字典——expires设置键的生存时间移除键的过期时间返回键的生存时间 Redis的过期删除策略1、定期删除2、惰性删除3、内存淘汰机制 过期键处理1、RDB功能对过期键的处理2、AOF功能对过期键的处理3、复制功能…

chatgpt赋能python:Python隐藏变量:探秘程序内部的“奥秘”

Python 隐藏变量&#xff1a;探秘程序内部的“奥秘” 作为一门高级编程语言&#xff0c;Python备受程序员的欢迎。但是&#xff0c;你是否了解Python隐藏变量&#xff1f;这些看似神秘的变量对程序员的编码工作有哪些影响呢&#xff1f;本篇文章将为您详细介绍Python隐藏变量的…