树(英语:tree)是一种抽象数据类型(ADT)或是实作这种抽象数据类型的数据结构,用来模拟具有树状结构性质的数据集合。它是由 nnn(n≥1n\geq 1n≥1)个有限节点组成一个具有层次关系的集合。把它叫做“树”是因为它看起来像一棵倒挂的树,也就是说它是根朝上,而叶朝下的。它具有以下的特点:
如下图,A 节点就是 B 节点的父节点,B 节点是 A 节点的子节点。B、C、D 这三个节点的父节点是同一个节点,所以它们之间互称为兄弟节点。我们把没有父节点的节点叫做根节点,也就是图中的节点 E。我们把没有子节点的节点叫做叶子节点或者叶节点,比如图中的 G、H、I、J、K、L 都是叶子节点。
用图片来展示什么是完全二叉树,请看下图:
下面这两棵树都是搜索树
最后一棵 不是平衡二叉树,因为它的左右两个子树的高度差的绝对值超过了1。
1. 链式存储法
一种基于指针或者引用的二叉链式存储法,每个节点有三个字段,其中一个存储数据,另外两个是指向左右子节点的指针。我们只要拎住根节点,就可以通过左右子节点的指针,把整棵树都串起来。这种存储方式我们比较常用。大部分二叉树代码都是通过这种结构来实现的。结构如下图:
2. 顺序存储法
顺序存储:将数据结构存储在固定的数组中,然在遍历速度上有一定的优势,但因所占空间比较大,是非主流二叉树。我们把根节点存储在下标 i = 1 的位置,那左子节点存储在下标 2 * i = 2 的位置,右子节点存储在 2 * i + 1 = 3 的位置。以此类推,B 节点的左子节点存储在 2 * i = 2 * 2 = 4 的位置,右子节点存储在 2 * i + 1 = 2 * 2 + 1 = 5 的位置。即如果节点 X 存储在数组中下标为 i 的位置,下标为 2 * i 的位置存储的就是左子节点,下标为 2 * i + 1 的位置存储的就是右子节点。
不过上图是一颗完全二叉树,所以数组仅仅浪费了下标为0的存储位置,如果是非完全二叉树,则可能会浪费比较多的数组内存空间。所以当要存储的树是一颗完全二叉树时,数组才是最合适的选择。所以,二叉树通常以链式存储。
二叉树是每个节点最多有两个子树的树结构。通常子树被称作“左子树”(left subtree)和“右子树”(right subtree)。
1.层结点
在二叉树的第 iii 层上最多有 2i−12^{i-1}2i−1 个结点(i>=1)
2.总结点
深度为 kkk 的二叉树最多有 2k+1−12^{k+1}-12k+1−1 个结点(k>=1)
3.深度
具有 nnn 个结点的完全二叉树的深度为 ⌊log2n⌋\lfloor \log_{2}n \rfloor⌊log2n⌋
4. 结点数
对于任意一棵二叉树,度为 0 的结点数等于度为 2 的结点数 +1。
5. 孩子结点
对完全二叉树,若从上至下、从左至右编号,则编号为 iii 的结点,其左孩子编号必为 2i2i2i,其右孩子编号必为 2i+12i+12i+1 ;其双亲的编号必为i/2(i=1 时为根,除外)
完全二叉树————若设二叉树的高度为hhh,除第 hhh 层外,其它各层 (1h−1)(1~h-1)(1 h−1) 的结点数都达到最大个数,第 hhh 层有叶子结点,并且叶子结点都是从左到右依次排布,这就是完全二叉树。
满二叉树——除了叶结点外每一个结点都有左右子叶且叶子结点都处在最底层的二叉树。
1. 列表实现
#! /user/bin/env python3
# -*- coding:utf-8 -*-
"""
@author: CarpeDiem
@date: 23/2/27
@version: 0.1
@description: 树的列表实现
"""def binary_tree(r):return [r, [], []]def insert_left(root, new_branch):t = root.pop(1) # 取出左子树if len(t) > 1: # 左子树已存在root.insert(1, [new_branch, t, []])else:root.insert(1, [new_branch, [], []])return rootdef insert_right(root, new_branch):t = root.pop(2) # 取出右子树if len(t) > 1: # 右子树已存在root.insert(2, [new_branch, [], t])else:root.insert(2, [new_branch, [], []])return rootdef get_root_val(root):return root[0]def set_root_val(root, new_val):root[0] = new_valdef get_left_child(root):return root[1]def get_right_child(root):return root[2]r = binary_tree(3)
insert_left(r, 4)
insert_left(r, 5)
insert_right(r, 6)
insert_right(r, 7)
print(r)
l = get_left_child(r)
print(l)
set_root_val(l, 9)
print(r)
insert_left(l, 11)
print(r)
print(get_right_child(get_right_child(r)))
2. 链表实现
class BinaryTree:def __init__(self, root_obj):self.key = root_objself.left_child = Noneself.right_child = Nonedef insert_left(self, new_node):if self.left_child == None:self.left_child = BinaryTree(new_node)else:t = BinaryTree(new_node)t.left_child = self.left_childself.left_child = tdef insert_right(self, new_node):if self.right_child == None:self.right_child = BinaryTree(new_node)else:t = BinaryTree(new_node)t.right_child = self.right_childself.right_child = tdef get_right_child(self):return self.right_childdef get_left_child(self):return self.left_childdef set_root_val(self, obj):self.key = objdef get_root_val(self):return self.key
二叉树主要有两种遍历方式:
从深度优先遍历和广度优先遍历进一步拓展,才有如下遍历方式:
1. 递归遍历
def pre_order(tree):if tree:print(tree.key)pre_order(tree.get_left_child())pre_order(tree.get_right_child())
1. 递归遍历
def in_order(tree):if tree:in_order(tree.get_left_child())print(tree.key)in_order(tree.get_right_child())
1. 递归遍历
def post_order(tree):if tree:post_order(tree.get_left_child())post_order(tree.get_right_child())
二叉树还没学完,后面在学习中持续补充,谢谢大家的鼓励和支持!
上一篇:锁消除、锁粗化、偏向锁、适应性锁