关于树你需要了解的一些概念:
这是一棵普通的树:
有节点,节点中有值,节点有指向下一个节点,节点还有度,节点还有层次(也叫高度或者深度)
结点的度
结点拥有的子树数目称为结点的度。
节点关系
结点子树的根结点为该结点的孩子结点。相应该结点称为孩子结点的双亲结点。
图2.2中,A为B的双亲结点,B为A的孩子结点。
同一个双亲结点的孩子结点之间互称兄弟结点。
图2.2中,结点B与结点C互为兄弟结点。
后面接的全是空的,我们叫它们叶子结点如:G、H、I
节点的层次:
从根开始定义起,根为第一层,根的孩子为第二层,以此类推。
树中结点的最大层次数称为树的深度或高度。图2.1所示树的深度为4。
二叉树
由二叉树定义以及图示分析得出二叉树有以下特点:
1)每个结点最多有两颗子树,所以二叉树中不存在度大于2的结点。
2)左子树和右子树是有顺序的,次序不能任意颠倒。
3)即使树中某结点只有一棵子树,也要区分它是左子树还是右子树。
简单来说就是:最多只有两个字树的树就叫二叉树
在日常编程中,二叉树有一些性质我们可能可以用到,比如
1)在二叉树的第i层上最多有2i-1次方 个节点 。(i>=1)
2)二叉树中如果深度为k,那么最多有2k次方-1个节点。(k>=1)
3)n0=n2+1 n0表示度数为0的节点数,n2表示度数为2的节点数。
4)在完全二叉树中,具有n个节点的完全二叉树的深度为[log2n]+1,其中[log2n]是向下取整。
5)若对含 n 个结点的完全二叉树从上到下且从左至右进行 1 至 n 的编号,则对完全二叉树中任意一个编号为 i 的结点有如下特性:
(1) 若 i=1,则该结点是二叉树的根,无双亲, 否则,编号为 [i/2] 的结点为其双亲结点;
(2) 若 2i>n,则该结点无左孩子, 否则,编号为 2i 的结点为其左孩子结点;
(3) 若 2i+1>n,则该结点无右孩子结点, 否则,编号为2i+1 的结点为其右孩子结点。
对于第五点我们来详细讲下并给出一些实例,因为这个笔试题中经常用到:
用数组顺序存储二叉树, 其显著特征是: (n表示数组的索引值)
-
- 在数组中index=n的元素对应的父节点为 index=(n-1)/2的元素, 向下取整;
-
- 在数组中index=n的元素对应的左子节点为 index=2*n+1的元素;
-
- 在数组中index=n的元素对应的右子节点为 index=2*n+2的元素;
如果此时给我们一个顺序存储的数组,叫我们构建一颗二叉树
给定二叉树 [3,9,20,null,null,15,7]
,
Definition for a binary tree node.
public class TreeNode {
int val;
TreeNode left;
TreeNode right;
TreeNode(int x) { val = x; }
}
public static TreeNode createBT(int[] arr, int i) // 初始时,传入的i==0
{
TreeNode root = null; // 定义根节点
if (i >= arr.length) // i >= arr.length 时,表示已经到达了根节点
return null;
root = new TreeNode(arr[i]); // 根节点
root.left = createBT(arr, 2*i+1); // 递归建立左孩子结点
root.right = createBT(arr, 2*i+2); // 递归建立右孩子结点
return r