二叉树(Binary Tree)

二叉树(Binary Tree)是一种树形数据结构,由节点构成,每个节点最多有两个子节点:一个左子节点和一个右子节点。

	public class TreeNode {
int val;
TreeNode left;
TreeNode right;
TreeNode(int val) { this.val = val; }
}

基本概念

"二叉树"(Binary Tree)这个名称的由来是因为二叉树的每个节点最多有两个子节点,一个左子节点和一个右子节点。其中,“二叉”指的是两个,因此“二叉树”表示每个节点最多可以分支成两个子节点。基本定义:

  • 每个节点包含一个值(或数据),另外最多有两个子节点。
  • 左子节点和右子节点的顺序是固定的,左边的子节点是左子节点,右边的子节点是右子节点。
  • 一个节点可以没有子节点(叶节点),也可以有一个子节点或两个子节点(内部节点)。

相关基本概念:

  • 根节点(Root): 二叉树的顶部节点称为根节点,是树的起始点
  • 节点(Node): 二叉树的基本构建单元。每个节点包含一个值(或数据)以及指向左子节点和右子节点的指针。
  • 父节点(Parent Node): 一个节点的直接上级节点,如果存在的话。例如,一个节点的左子节点的父节点是该节点本身。
  • 叶节点(Leaf Node): 没有子节点的节点称为叶节点,即左子节点和右子节点都为空。
  • 子树(Subtree): 以某个节点为根的树,它包括该节点及其所有后代节点。
  • 高度(Height): 从某个节点到其最远叶节点的最长路径上的边数,也称为节点的层数。叶节点的高度为0。

在二叉树基本定义上,加上一些规则,可以衍生出更多种类的二叉树。比如:

二叉搜索树(Binary Search Tree,BST): 一种特殊的二叉树,满足以下性质:对于树中的每个节点,其左子树中的值都小于该节点的值,而其右子树中的值都大于该节点的值。BST通常用于实现有序数据集合。

完全二叉树(Complete Binary Tree): 一个二叉树,其所有层次(深度)除了最后一层外,都是完全填充的,且最后一层的节点从左到右填充,没有空隙。

平衡二叉树(Balanced Binary Tree): 一种高度平衡的二叉树,其中每个节点的两棵子树的高度差不超过1。平衡二叉树通常用于提高查找、插入和删除操作的性能。

预备基础算法 —— 递归(Recursion)

下一部分要写的是二叉树基本遍历代码实现其实可以有多种,思量后用递归实现应该是初接触者比较简洁好理解的方式。为此,在写二叉树下一部分内容之前简单写下基础递归算法,以保证本系列文章承前启后。

递归(Recursion),在数学与计算机科学中对其描述的说法有很多,比如:

  1. 指在函数的定义中使用函数自身的方法;
  2. 指一种通过重复将问题分解为同类的子问题而解决问题的方法;

    (PS:这里同类子问题对于于上一种说法就是函数自身)
  3. 指由一种(或多种)简单的基本情况定义的一类对象或方法,并规定其他所有情况都能被还原为其基本情况。

    (PS:这里描述的基本情况对应于第一种说法中的函数自身了)

当然本文非学术著作"哪种描述比较合适"在此不多做分析,从编码实践的角度第一种说法更为地气一点。

“将问题分解为同类的子问题” 这一点是用递归的方式来解题的关键,这里用个简单的累加和的例子:

设计一个函数,输入参数为 n ,返回 1+2+...n 的和。

public int sum(int n){
int result = 0;
for (int i = 1; i <= n ; i++) {
result = result + i;
}
return result;
}

想想初学 C 语言for循环的时候应该都有写过上述代码,从 1开始递增加到 n 这其实是典型的递推。那如果用递归的思路来思考的话:

求 1+2+..n 的和(问题) -> 就是 n 加上 求 1+2+..(n-1) 的和(同类的子问题);

其中最基本的情况 1 的和 为 1。

public int sum( int n ) {
if( 1 == n) return 1;
return n + sum(n-1);
}

可以看到递归的代码实现上是不是非常简洁。大部分初学者思考上比较习惯于递推,如果第一次接触递归角度思考会有些不适应(或者无法独立分析出来递归)也是正常。当慢慢熟悉后,会发现用递归的思路解决某些算法问题往往会非常简单(在本篇接下来的内容中就能发现这点)。

在 初学递归 过度到 熟悉递归 这个阶段,笔者建议可以考虑把一些用递推已经解决了的问题 用 递归的思路尝试解决,习惯递归思路后会打开一片新世界。

基本遍历(Traversal)

二叉树的遍历是指按照一定的顺序访问二叉树的所有节点。在二叉树中,有三种常见的遍历方式,它们分别是前序遍历、中序遍历和后序遍历。

先序遍历(Preorder Traversal)

从根节点开始,首先访问根节点,然后按照前序遍历的方式依次访问左子树和右子树。前序遍历通常用于复制一棵树或计算表达式的值。

访问顺序:根节点 -> 左子树 -> 右子树

Leetcode 144. 二叉树的前序遍历【简单】

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

中序遍历(Inorder Traversal)

从根节点开始,首先按照中序遍历的方式访问左子树,然后访问根节点,最后访问右子树。中序遍历通常用于访问二叉搜索树中的节点,以升序或降序访问节点值。

访问顺序:左子树 -> 根节点 -> 右子树

Leetcode 94. 二叉树的中遍历【简单】

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

针对后序遍历(Postorder Traversal)从根节点开始,首先按照后序遍历的方式访问左子树,然后访问右子树,最后访问根节点。后序遍历通常用于释放二叉树的内存,或计算表达式的值。访问顺序:左子树 -> 右子树 -> 根节点,在此不过多描述相信一定能够完成编码。

反向构建

Leetcode 105. 从前序与中序遍历序列构造二叉树【中等】

给定两个整数数组 preorder 和 inorder ,其中 preorder 是二叉树的先序遍历, inorder 是同一棵树的中序遍历,请构造二叉树并返回其根节点。

综合应用

本系列文章中已经介绍了链表、递归、二叉树,解决算法问题往往会需要综合应用。不妨来看下下面这个问题:

Leetcode 114. 二叉树展开为链表【中等】

给你二叉树的根结点 root ,请你将它展开为一个单链表:

展开后的单链表应该同样使用 TreeNode ,其中 right 子指针指向链表中下一个结点,而左子指针始终为 null 。

展开后的单链表应该与二叉树 先序遍历 顺序相同。



总结下

  • 介绍了二叉树的的一些基本概念包括:根节点、叶子节点、高度等等;
  • 介绍了基础算法递归的思想:“重复将问题分解为同类的子问题而解决问题的方法”;
  • 介绍了基本的二叉树遍历 和 反向构建的相关思路;
  • 结合本系列先前文章内容,解决综合链表、递归、二叉树的问题,灵活处理使用数据结构的特征是关键。

数据结构与算法 | 二叉树(Binary Tree)的更多相关文章

  1. Java数据结构和算法 - 二叉树

    前言 数据结构可划分为线性结构.树型结构和图型结构三大类.前面几篇讨论了数组.栈和队列.链表都是线性结构.树型结构中每个结点只允许有一个直接前驱结点,但允许有一个以上直接后驱结点.树型结构有树和二叉树 ...

  2. javascript数据结构与算法-- 二叉树

    javascript数据结构与算法-- 二叉树 树是计算机科学中经常用到的一种数据结构.树是一种非线性的数据结构,以分成的方式存储数据,树被用来存储具有层级关系的数据,比如文件系统的文件,树还被用来存 ...

  3. javascript数据结构与算法---二叉树(删除节点)

    javascript数据结构与算法---二叉树(删除节点) function Node(data,left,right) { this.data = data; this.left = left; t ...

  4. javascript数据结构与算法---二叉树(查找最小值、最大值、给定值)

    javascript数据结构与算法---二叉树(查找最小值.最大值.给定值) function Node(data,left,right) { this.data = data; this.left ...

  5. javascript数据结构与算法--二叉树遍历(后序)

    javascript数据结构与算法--二叉树遍历(后序) 后序遍历先访问叶子节点,从左子树到右子树,再到根节点. /* *二叉树中,相对较小的值保存在左节点上,较大的值保存在右节点中 * * * */ ...

  6. javascript数据结构与算法--二叉树遍历(先序)

    javascript数据结构与算法--二叉树遍历(先序) 先序遍历先访问根节点, 然后以同样方式访问左子树和右子树 代码如下: /* *二叉树中,相对较小的值保存在左节点上,较大的值保存在右节点中 * ...

  7. javascript数据结构与算法--二叉树遍历(中序)

    javascript数据结构与算法--二叉树遍历(中序) 中序遍历按照节点上的键值,以升序访问BST上的所有节点 代码如下: /* *二叉树中,相对较小的值保存在左节点上,较大的值保存在右节点中 * ...

  8. 算法与数据结构基础 - 二叉树(Binary Tree)

    二叉树基础 满足这样性质的树称为二叉树:空树或节点最多有两个子树,称为左子树.右子树, 左右子树节点同样最多有两个子树. 二叉树是递归定义的,因而常用递归/DFS的思想处理二叉树相关问题,例如Leet ...

  9. [算法专题] Binary Tree

    1 Same Tree https://leetcode.com/problems/same-tree/ Given two binary trees, write a function to che ...

  10. [Swift]LeetCode968.监控二叉树 | Binary Tree Cameras

    Given a binary tree, we install cameras on the nodes of the tree. Each camera at a node can monitor  ...

随机推荐

  1. PostgreSQL 性能优化: 等待事件

    等待事件 等待事件是 PostgreSQL 的重要优化工具.当您能查明会话为什么在等待资源以及会话在做什么时,您就能更好地减少瓶颈.您可以使用本节中的信息来查找可能的原因和纠正措施. 目录 等待事件概 ...

  2. 树莓派4B-细分驱动步进电机

    树莓派4B-细分驱动步进电机 项目介绍 利用4B树莓派控制步进电机转动,精度可达:0.0144度 (即360度/25000) 适用于非常精密的角度转动. 舵机的精度为1度,无法实现超高精度控制. 硬件 ...

  3. Parallel 与 ConcurrentBag<T> 这对儿黄金搭档

    〇.前言 日常开发中经常会遇到数据统计,特别是关于报表的项目.数据处理的效率和准确度当然是首要关注点. 本文主要介绍,如何通过 Parallel 来并行处理数据,并组合 ConcurrentBag&l ...

  4. Python单元测试之道:从入门到精通的全面指南

    在这篇文章中,我们会深入探讨Python单元测试的各个方面,包括它的基本概念.基础知识.实践方法.高级话题,如何在实际项目中进行单元测试,单元测试的最佳实践,以及一些有用的工具和资源 一.单元测试重要 ...

  5. 我真的,AI框架的编程范式怎么理解?

    我给领导汇报AI框架用函数式编程好,没讲明白,说函数式就是写函数那样方便,都被领导吊飞了,啥玩意,写啥不是写函数,狗屁不通! 网上搜说用tensorflow那就是用声明式编程,用pytorch就是命令 ...

  6. 2021-7-7 VUE笔记2

    if实例 <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <scri ...

  7. Confluence 挖矿病毒 升级现有系统

    Confluence 挖矿病毒 升级现有系统 背景 服务器很多服务都很卡,通过检查发现是一起运行的confluence异常,被挖矿病毒挖矿,华为云和官网也有说明. 知道问题之后,处理方式就是将现有的问 ...

  8. Python操纵数据库:peewee

    模块名:peewee 安装方法: pip install peewee 官方教程:http://docs.peewee-orm.com/en/latest/

  9. JavaWeb和MVC三层架构

    JavaWeb 概述 网站发布和部署一定要依托技术语言吗: 不一定,一个网站可以直接发布和部署,因为因为浏览器能够识别网页只需要两样东西,网络和静态页面,还有一个装在他们的容器,比如 nginx. 静 ...

  10. 如何通过cookie、session鉴权(nodejs/koa)

    http是一种无状态的协议,每一个请求都是独立的,即使同一个页面向服务器发送多次请求,服务器也无法区分是不是同一用户,所以这个时候可以借助于cookie来做身份认证,当用户登录成功,服务器为浏览器设置 ...