• 一、树的介绍
    • 1.树的定义
    • 2.树的基本术语
  • 二、二叉树的介绍
    • 1.二叉树的定义
    • 2. 二叉树的性质
    • 3. 满二叉树,完全二叉树和二叉查找树
  • 三、二叉查找树的Java实现
    • 1.二叉查找树节点的定义
    • 2.遍历
    • 3. 查找
    • 4. 最大值和最小值
    • 5. 前驱和后继
    • 6. 插入
    • 7. 删除
    • 8. 打印
    • 9. 销毁
  • 四、树的深度/广度优先遍历

    一、树的介绍

    1.树的定义

    树是一种数据结构,它是由n(n>=1)个有限节点组成一个具有层次关系的集合。

    img

    把它叫做“树”是因为它看起来像一棵倒挂的树,也就是说它是根朝上,而叶朝下的。它具有以下的特点:

    (01) 每个节点有零个或多个子节点;

    (02) 没有父节点的节点称为根节点;

    (03) 每一个非根节点有且只有一个父节点;

    (04) 除了根节点外,每个子节点可以分为多个不相交的子树。

    2.树的基本术语

    若一个结点有子树,那么该结点称为子树根的”双亲”,子树的根是该结点的”孩子”。有相同双亲的结点互为”兄弟”。一个结点的所有子树上的任何结点都是该结点的后裔。从根结点到某个结点的路径上的所有结点都是该结点的祖先。

    结点的度:结点拥有的子树的数目。

    叶子:度为零的结点。

    分支结点:度不为零的结点。

    树的度:树中结点的最大的度。

    层次:根结点的层次为1,其余结点的层次等于该结点的双亲结点的层次加1。

    树的高度:树中结点的最大层次。

    无序树:如果树中结点的各子树之间的次序是不重要的,可以交换位置。

    有序树:如果树中结点的各子树之间的次序是重要的, 不可以交换位置。

    森林:0个或多个不相交的树组成。对森林加上一个根,森林即成为树;删去根,树即成为森林。

    二、二叉树的介绍

    1.二叉树的定义

    二叉树是每个节点最多有两个子树的树结构。它有五种基本形态:二叉树可以是空集;根可以有空的左子树或右子树;或者左、右子树皆为空。

    img

    2. 二叉树的性质

    二叉树有以下几个性质:

    性质1:二叉树第i层上的结点数目最多为$2^{i-1}$ (i≥1)。

    性质2:深度为k的二叉树至多有$2^k - 1$个结点(k≥1)。

    性质3:包含n个结点的二叉树的高度至少为$log_2(n+1)$。

    性质4:在任意一棵二叉树中,若叶子结点的个数为n0,度为2的结点数为n2,则n0=n2+1

    2.1 性质1:二叉树第i层上的结点数目最多为$2^{i-1}$(i≥1)

    证明:下面用”数学归纳法”进行证明。

    ​ (01) 当i=1时,第i层的节点数目为1。因为第1层上只有一个根结点,所以命题成立。

    ​ (02) 假设当i>1,第i层的节点数目为$2^{i-1}$。这个是根据(01)推断出来的!

    下面根据这个假设,推断出”第(i+1)层的节点数目为$2^{i}$”即可。

    由于二叉树的每个结点至多有两个孩子,故”第(i+1)层上的结点数目” 最多是 “第i层的结点数目的2倍”。即,第(i+1)层上的结点数目最大值=2×$2^{i-1}$=$2^{i}$。

    故假设成立,原命题得证!

    2.2 性质2:深度为k的二叉树至多有$2^{k}-1$个结点(k≥1)

    证明:在具有相同深度的二叉树中,当每一层都含有最大结点数时,其树中结点数最多。利用”性质1”可知,深度为k的二叉树的结点数至多为:

    ​ $ 2^0+2^1+…+2^{k-1}=2^k-1$

    故原命题得证!

    2.3 性质3:包含n个结点的二叉树的高度至少为$log_2(n+1)$

    证明:根据”性质2”可知,高度为h的二叉树最多有$2^{h}-1$个结点。反之,对于包含n个节点的二叉树的高度至少为$log_2(n+1)$。

    2.4 性质4:在任意一棵二叉树中,若终端结点的个数为n0,度为2的结点数为n2,则n0=n2+1

    证明:因为二叉树中所有结点的度数均不大于2,所以结点总数(记为n)=”0度结点数(n0)” + “1度结点数(n1)” + “2度结点数(n2)”。由此,得到等式一。

    (等式一) n=n0+n1+n2

    另一方面,0度结点没有孩子,1度结点有一个孩子,2度结点有两个孩子,故二叉树中孩子结点总数是:n1+2n2。此外,只有根不是任何结点的孩子。故二叉树中的结点总数又可表示为等式二。

    (等式二) n=n1+2n2+1

    由(等式一)和(等式二)计算得到:n0=n2+1。原命题得证!

    3. 满二叉树,完全二叉树和二叉查找树

    3.1 满二叉树

    定义:高度为h,并且由$2^{h}-1$个结点的二叉树,被称为满二叉树。

    img

    3.2 完全二叉树

    定义:一棵二叉树中,只有最下面两层结点的度可以小于2,并且最下一层的叶结点集中在靠左的若干位置上。这样的二叉树称为完全二叉树。

    特点:叶子结点只能出现在最下层和次下层,且最下层的叶子结点集中在树的左部。显然,一棵满二叉树必定是一棵完全二叉树,而完全二叉树未必是满二叉树。

    img

    3.3 二叉查找树

    定义:二叉查找树(Binary Search Tree),又被称为二叉搜索树。设x为二叉查找树中的一个结点,x节点包含关键字key,节点x的key值记为key[x]。如果y是x的左子树中的一个结点,则key[y] <= key[x];如果y是x的右子树的一个结点,则key[y] >= key[x]。

    img

    在二叉查找树中:

    (01) 若任意节点的左子树不空,则左子树上所有结点的值均小于它的根结点的值;

    (02) 任意节点的右子树不空,则右子树上所有结点的值均大于它的根结点的值;

    (03) 任意节点的左、右子树也分别为二叉查找树。

    (04) 没有键值相等的节点(no duplicate nodes)。

    三、二叉查找树的Java实现

    1.二叉查找树节点的定义

    1. public class BSTree<T extends Comparable<T>> {
    2. private BSTNode<T> mRoot; // 根结点
    3. public class BSTNode<T extends Comparable<T>> {
    4. T key; // 关键字(键值)
    5. BSTNode<T> left; // 左孩子
    6. BSTNode<T> right; // 右孩子
    7. BSTNode<T> parent; // 父结点
    8. public BSTNode(T key, BSTNode<T> parent, BSTNode<T> left, BSTNode<T> right) {
    9. this.key = key;
    10. this.parent = parent;
    11. this.left = left;
    12. this.right = right;
    13. }
    14. }
    15. ......
    16. }

    BSTree是二叉树,它保含了二叉树的根节点mRoot;mRoot是BSTNode类型,而BSTNode是二叉查找树的节点,它是BSTree的内部类。BSTNode包含二叉查找树的几个基本信息:

    (01) key — 它是关键字,是用来对二叉查找树的节点进行排序的。

    (02) left — 它指向当前节点的左孩子。

    (03) right — 它指向当前节点的右孩子。

    (04) parent — 它指向当前节点的父结点。

    2.遍历

    这里讲解前序遍历、中序遍历、后序遍历3种方式。

    2.1 前序遍历

    若二叉树非空,则执行以下操作:

    (01) 访问根结点;

    (02) 先序遍历左子树;

    (03) 先序遍历右子树。

    前序遍历代码

    1. private void preOrder(BSTNode<T> tree) {
    2. if(tree != null) {
    3. System.out.print(tree.key+" ");
    4. preOrder(tree.left);
    5. preOrder(tree.right);
    6. }
    7. }
    8. public void preOrder() {
    9. preOrder(mRoot);
    10. }

    2.2 中序遍历

    若二叉树非空,则执行以下操作:

    (01) 中序遍历左子树;

    (02) 访问根结点;

    (03) 中序遍历右子树。

    中序遍历代码

    1. private void inOrder(BSTNode<T> tree) {
    2. if(tree != null) {
    3. inOrder(tree.left);
    4. System.out.print(tree.key+" ");
    5. inOrder(tree.right);
    6. }
    7. }
    8. public void inOrder() {
    9. inOrder(mRoot);
    10. }

    2.3 后序遍历

    若二叉树非空,则执行以下操作:

    (01) 后序遍历左子树;

    (02) 后序遍历右子树;

    (03) 访问根结点。

    后序遍历代码

    1. private void postOrder(BSTNode<T> tree) {
    2. if(tree != null)
    3. {
    4. postOrder(tree.left);
    5. postOrder(tree.right);
    6. System.out.print(tree.key+" ");
    7. }
    8. }
    9. public void postOrder() {
    10. postOrder(mRoot);
    11. }

    看看下面这颗树的各种遍历方式:

    img

    对于上面的二叉树而言,

    (01) 前序遍历结果: 3 1 2 5 4 6

    (02) 中序遍历结果: 1 2 3 4 5 6

    (03) 后序遍历结果: 2 1 4 6 5 3

    3. 查找

    递归版本的代码

    1. /*
    2. * (递归实现)查找"二叉树x"中键值为key的节点
    3. */
    4. private BSTNode<T> search(BSTNode<T> x, T key) {
    5. if (x==null)
    6. return x;
    7. int cmp = key.compareTo(x.key);
    8. if (cmp < 0)
    9. return search(x.left, key);
    10. else if (cmp > 0)
    11. return search(x.right, key);
    12. else
    13. return x;
    14. }
    15. public BSTNode<T> search(T key) {
    16. return search(mRoot, key);
    17. }

    非递归版本的代码

    1. /*
    2. * (非递归实现)查找"二叉树x"中键值为key的节点
    3. */
    4. private BSTNode<T> iterativeSearch(BSTNode<T> x, T key) {
    5. while (x!=null) {
    6. int cmp = key.compareTo(x.key);
    7. if (cmp < 0)
    8. x = x.left;
    9. else if (cmp > 0)
    10. x = x.right;
    11. else
    12. return x;
    13. }
    14. return x;
    15. }
    16. public BSTNode<T> iterativeSearch(T key) {
    17. return iterativeSearch(mRoot, key);
    18. }

    4. 最大值和最小值

    查找最大值的代码

    1. /*
    2. * 查找最大结点:返回tree为根结点的二叉树的最大结点。
    3. */
    4. private BSTNode<T> maximum(BSTNode<T> tree) {
    5. if (tree == null)
    6. return null;
    7. while(tree.right != null)
    8. tree = tree.right;
    9. return tree;
    10. }
    11. public T maximum() {
    12. BSTNode<T> p = maximum(mRoot);
    13. if (p != null)
    14. return p.key;
    15. return null;
    16. }

    查找最小值的代码

    1. /*
    2. * 查找最小结点:返回tree为根结点的二叉树的最小结点。
    3. */
    4. private BSTNode<T> minimum(BSTNode<T> tree) {
    5. if (tree == null)
    6. return null;
    7. while(tree.left != null)
    8. tree = tree.left;
    9. return tree;
    10. }
    11. public T minimum() {
    12. BSTNode<T> p = minimum(mRoot);
    13. if (p != null)
    14. return p.key;
    15. return null;
    16. }

    5. 前驱和后继

    节点的前驱:是该节点的左子树中的最大节点。

    节点的后继:是该节点的右子树中的最小节点。

    查找前驱节点的代码

    1. /*
    2. * 找结点(x)的前驱结点。即,查找"二叉树中数据值小于该结点"的"最大结点"。
    3. */
    4. public BSTNode<T> predecessor(BSTNode<T> x) {
    5. // 如果x存在左孩子,则"x的前驱结点"为 "以其左孩子为根的子树的最大结点"。
    6. if (x.left != null)
    7. return maximum(x.left);
    8. // 如果x没有左孩子。则x有以下两种可能:
    9. // (01) x是"一个右孩子",则"x的前驱结点"为 "它的父结点"。
    10. // (02) x是"一个左孩子",则查找"x的最低的父结点,并且该父结点要具有右孩子",找到的这个"最低的父结点"就是"x的前驱结点"。
    11. BSTNode<T> y = x.parent;
    12. while ((y!=null) && (x==y.left)) {//满足条件,不断往上追溯,直到找到右祖先结点
    13. x = y;
    14. y = y.parent;
    15. }
    16. return y;
    17. }

    查找后继节点的代码

    1. /*
    2. * 找结点(x)的后继结点。即,查找"二叉树中数据值大于该结点"的"最小结点"。
    3. */
    4. public BSTNode<T> successor(BSTNode<T> x) {
    5. // 如果x存在右孩子,则"x的后继结点"为 "以其右孩子为根的子树的最小结点"。
    6. if (x.right != null)
    7. return minimum(x.right);
    8. // 如果x没有右孩子。则x有以下两种可能:
    9. // (01) x是"一个左孩子",则"x的后继结点"为 "它的父结点"。
    10. // (02) x是"一个右孩子",则查找"x的最低的父结点,并且该父结点要具有左孩子",找到的这个"最低的父结点"就是"x的后继结点"。
    11. BSTNode<T> y = x.parent;
    12. while ((y!=null) && (x==y.right)) {//满足条件,不断往上追溯,直到找到右祖先结点
    13. x = y;
    14. y = y.parent;
    15. }
    16. return y;
    17. }

    6. 插入

    插入节点的代码

    1. /*
    2. * 将结点插入到二叉树中
    3. *
    4. * 参数说明:
    5. * tree 二叉树的
    6. * z 插入的结点
    7. */
    8. private void insert(BSTree<T> bst, BSTNode<T> z) {
    9. int cmp;
    10. BSTNode<T> y = null;
    11. BSTNode<T> x = bst.mRoot;
    12. // 查找z的插入位置
    13. while (x != null) {
    14. y = x;
    15. cmp = z.key.compareTo(x.key);
    16. if (cmp < 0)
    17. x = x.left;
    18. else
    19. x = x.right;
    20. }
    21. z.parent = y;
    22. if (y==null)
    23. bst.mRoot = z;
    24. else {
    25. cmp = z.key.compareTo(y.key);
    26. if (cmp < 0)
    27. y.left = z;
    28. else
    29. y.right = z;
    30. }
    31. }
    32. /*
    33. * 新建结点(key),并将其插入到二叉树中
    34. *
    35. * 参数说明:
    36. * tree 二叉树的根结点
    37. * key 插入结点的键值
    38. */
    39. public void insert(T key) {
    40. BSTNode<T> z=new BSTNode<T>(key,null,null,null);
    41. // 如果新建结点失败,则返回。
    42. if (z != null)
    43. insert(this, z);
    44. }

    注:本文实现的二叉查找树是允许插入相同键值的节点的。

    7. 删除

    删除节点的代码

    1. /*
    2. * 删除结点(z),并返回被删除的结点
    3. *
    4. * 参数说明:
    5. * bst 二叉树
    6. * z 删除的结点
    7. */
    8. private BSTNode<T> remove(BSTree<T> bst, BSTNode<T> z) {
    9. BSTNode<T> x=null;
    10. BSTNode<T> y=null;
    11. if ((z.left == null) || (z.right == null) )
    12. y = z;
    13. else
    14. y = successor(z);
    15. if (y.left != null)
    16. x = y.left;
    17. else
    18. x = y.right;
    19. if (x != null)
    20. x.parent = y.parent;
    21. if (y.parent == null)
    22. bst.mRoot = x;
    23. else if (y == y.parent.left)
    24. y.parent.left = x;
    25. else
    26. y.parent.right = x;
    27. if (y != z)
    28. z.key = y.key;
    29. return y;
    30. }
    31. /*
    32. * 删除结点(z),并返回被删除的结点
    33. *
    34. * 参数说明:
    35. * tree 二叉树的根结点
    36. * z 删除的结点
    37. */
    38. public void remove(T key) {
    39. BSTNode<T> z, node;
    40. if ((z = search(mRoot, key)) != null)
    41. if ( (node = remove(this, z)) != null)
    42. node = null;
    43. }

    8. 打印

    打印二叉查找树的代码

    1. /*
    2. * 打印"二叉查找树"
    3. *
    4. * key -- 节点的键值
    5. * direction -- 0,表示该节点是根节点;
    6. * -1,表示该节点是它的父结点的左孩子;
    7. * 1,表示该节点是它的父结点的右孩子。
    8. */
    9. private void print(BSTNode<T> tree, T key, int direction) {
    10. if(tree != null) {
    11. if(direction==0) // tree是根节点
    12. System.out.printf("%2d is root\n", tree.key);
    13. else // tree是分支节点
    14. System.out.printf("%2d is %2d's %6s child\n", tree.key, key, direction==1?"right" : "left");
    15. print(tree.left, tree.key, -1);
    16. print(tree.right,tree.key, 1);
    17. }
    18. }
    19. public void print() {
    20. if (mRoot != null)
    21. print(mRoot, mRoot.key, 0);
    22. }

    9. 销毁

    销毁二叉查找树的代码

    1. /*
    2. * 销毁二叉树
    3. */
    4. private void destroy(BSTNode<T> tree) {
    5. if (tree==null)
    6. return ;
    7. if (tree.left != null)
    8. destroy(tree.left);
    9. if (tree.right != null)
    10. destroy(tree.right);
    11. tree=null;
    12. }
    13. public void clear() {
    14. destroy(mRoot);
    15. mRoot = null;
    16. }

    四、树的深度/广度优先遍历

    树的深度优先遍历需要用到额外的数据结构—->栈;而广度优先遍历需要队列来辅助;这里以二叉树为例来实现。

    1. import java.util.ArrayDeque;
    2. public class BinaryTree {
    3. static class TreeNode{
    4. int value;
    5. TreeNode left;
    6. TreeNode right;
    7. public TreeNode(int value){
    8. this.value=value;
    9. }
    10. }
    11. TreeNode root;
    12. public BinaryTree(int[] array){
    13. root=makeBinaryTreeByArray(array,1);
    14. }
    15. /**
    16. * 采用递归的方式创建一颗二叉树
    17. * 传入的是二叉树的数组表示法
    18. * 构造后是二叉树的二叉链表表示法
    19. */
    20. public static TreeNode makeBinaryTreeByArray(int[] array,int index){
    21. if(index<array.length){
    22. int value=array[index];
    23. if(value!=0){
    24. TreeNode t=new TreeNode(value);
    25. array[index]=0;
    26. t.left=makeBinaryTreeByArray(array,index*2);
    27. t.right=makeBinaryTreeByArray(array,index*2+1);
    28. return t;
    29. }
    30. }
    31. return null;
    32. }
    33. /**
    34. * 深度优先遍历,相当于先根遍历
    35. * 采用非递归实现
    36. * 需要辅助数据结构:栈
    37. */
    38. public void depthOrderTraversal(){
    39. if(root==null){
    40. System.out.println("empty tree");
    41. return;
    42. }
    43. ArrayDeque<TreeNode> stack=new ArrayDeque<TreeNode>();
    44. stack.push(root);
    45. while(stack.isEmpty()==false){
    46. TreeNode node=stack.pop();
    47. System.out.print(node.value+" ");
    48. if(node.right!=null){
    49. stack.push(node.right);
    50. }
    51. if(node.left!=null){
    52. stack.push(node.left);
    53. }
    54. }
    55. System.out.print("\n");
    56. }
    57. /**
    58. * 广度优先遍历
    59. * 采用非递归实现
    60. * 需要辅助数据结构:队列
    61. */
    62. public void levelOrderTraversal(){
    63. if(root==null){
    64. System.out.println("empty tree");
    65. return;
    66. }
    67. ArrayDeque<TreeNode> queue=new ArrayDeque<TreeNode>();
    68. queue.add(root);
    69. while(queue.isEmpty()==false){
    70. TreeNode node=queue.remove();
    71. System.out.print(node.value+" ");
    72. if(node.left!=null){
    73. queue.add(node.left);
    74. }
    75. if(node.right!=null){
    76. queue.add(node.right);
    77. }
    78. }
    79. System.out.print("\n");
    80. }
    81. /**
    82. * 13
    83. * / \
    84. * 65 5
    85. * / \ \
    86. * 97 25 37
    87. * / /\ /
    88. * 22 4 28 32
    89. */
    90. public static void main(String[] args) {
    91. int[] arr={0,13,65,5,97,25,0,37,22,0,4,28,0,0,32,0};
    92. BinaryTree tree=new BinaryTree(arr);
    93. tree.depthOrderTraversal();
    94. tree.levelOrderTraversal();
    95. }
    96. }