二叉树
一、概念
一棵二叉树是节点的有限集合,该集合可以为空,也可以是由一个根节点和左子树、右子树组成的二叉树。
二叉树的特点:
- 每个节点最多有两棵子树,即二叉树的度最大为2
- 二叉树是有序树,不是指排好序的树,而是二叉树有左右之分
二、二叉树的几种形态
从左到右依次为:空树、只有根节点的二叉树、只有左节点的二叉树、只有右节点的二叉树、带左右节点的二叉树
三、完全二叉树和满二叉树
满二叉树:如果一棵二叉树,它的每一层节点数都达到了该层节点数的最大值,那么这棵树就是一个满二叉树。即一棵满二叉树有n层,那么它的节点数为2^n - 1
个。
完全二叉树:对于深度为k,有n个节点的二叉树,当且仅当其每一个节点都与深度为k的满二叉树中编号从1到n的节点一一对应时称为完全二叉树。满二叉树是一种特殊的完全二叉树。
对于完全二叉树,如果一个节点有右孩子,那么它必须有左孩子,否则该树不能称作完全二叉树,下图就不是完全二叉树
四、二叉树的性质
- 如果规定根节点的层数为1,那么一棵非空二叉树的第i层上最多有
2^(i - 1)
个节点 - 如果固定根节点的深度为1,那么一颗深度为k的二叉树最多有
2^k - 1
个节点 - 有n个节点的完全二叉树的深度为log2(n + 1),上取整
- 任意一颗二叉树,如果叶子节点数为n0,度为2的非叶节点个数为n2,则
n0 = n2 + 1
- 对于有n个节点的完全二叉树,如果从上至下从左至右的顺序对每个节点从0开始编号,则对于序号为i的节点有
- 若i > 0,双亲序号:
(i - 1) / 2
;i = 0,i为根节点,无双亲 - 若
2i + 1 < n
,左孩子序号:2i + 1
,否则无左孩子 - 若
2i+ 1 < n
,右孩子序号:2i + 2
,否则无右孩子
- 若i > 0,双亲序号:
五、二叉树的遍历
二叉树的遍历顺序是固定的,不同的地方在于打印节点值的时机。
1、前序遍历 (根左右)
递归方法:
public void preorder(TreeNode root) {
if(root == null) return;
System.out.print(root.val); //每次遇到root先打印root的值
preorder(root.left); //遍历左子树
preorder(root.right);//遍历右子树
}
非递归方法:用栈实现
public void preorder(TreeNode root) {
if(root == null) return;
Stack<TreeNode> stack = new Stack<>();
TreeNode cur = root; //cur指向root
while(cur != null || !stack.isEmpty()){
//右子树为空,但stack不为空
while(cur != null) {
//cur不为空,则将cur入栈打印,cur = cur.left
stack.push(cur);
System.out.print(cur.val);
cur = cur.left
}
//cur为空时,pop出栈顶元素
TreeNode top = stack.pop();
cur = top.right; //遍历子树,右子树可能为空
}
}