计算彻底二叉树的节点数

基本概念

若是让你数一下一棵普通二叉树有多少个节点,这很简单,只要在二叉树的遍历框架上加一点代码就好了。java

可是,若是给你一棵彻底二叉树,让你计算它的节点个数,你会不会?算法的时间复杂度是多少?算法

这个算法的时间复杂度应该是 O(logN*logN),若是你心中的算法没有达到这么高效,那么本文就是给你写的。数组

首先要明确一下两个关于二叉树的名词「彻底二叉树」和「满二叉树」。markdown

咱们说的彻底二叉树以下图,每一层都是紧凑靠左排列的:框架

image.png

咱们说的满二叉树以下图,是一种特殊的彻底二叉树,每层都是是满的,像一个稳定的三角形:spa

image.png

说句题外话,关于这两个定义,中文语境和英文语境彷佛有点区别,咱们说的彻底二叉树对应英文 Complete Binary Tree,没有问题。可是咱们说的满二叉树对应英文 Perfect Binary Tree,而英文中的 Full Binary Tree 是指一棵二叉树的全部节点要么没有孩子节点,要么有两个孩子节点。以下:3d

image.png

以上定义出自 wikipedia,这里就是顺便一提,其实名词叫什么都无所谓,重要的是算法操做。code

记住「满二叉树」和「彻底二叉树」的区别,等会会用到orm

节点数目计算

如今回归正题,如何求一棵彻底二叉树的节点个数呢?递归

// 输入一棵彻底二叉树,返回节点总数
int countNodes(TreeNode root);
复制代码

若是是一个普通二叉树,显然只要向下面这样遍历一边便可,时间复杂度 O(N):

public int countNodes(TreeNode root) {
    if (root == nullreturn 0;
    return 1 + countNodes(root.left) + countNodes(root.right);
}
复制代码

那若是是一棵二叉树,节点总数就和树的高度呈指数关系,时间复杂度 O(logN):

public int countNodes(TreeNode root) {
    int h = 0;
    // 计算树的高度
    while (root != null) {
        root = root.left;
        h++;
    }
    // 节点总数就是 2^h - 1
    return (int)Math.pow(2, h) - 1;
}
复制代码

彻底二叉树比普通二叉树特殊,但又没有满二叉树那么特殊,计算它的节点总数,能够说是普通二叉树和彻底二叉树的结合版,先看代码:

public int countNodes(TreeNode root) {
    TreeNode l = root, r = root;
    // 记录左、右子树的高度
    int hl = 0, hr = 0;
    while (l != null) {
        l = l.left;
        hl++;
    }
    while (r != null) {
        r = r.right;
        hr++;
    }
    // 若是左右子树的高度相同,则是一棵满二叉树
    if (hl == hr) {
        return (int)Math.pow(2, hl) - 1;
    }
    // 若是左右高度不一样,则按照普通二叉树的逻辑计算
    return 1 + countNodes(root.left) + countNodes(root.right);
}
复制代码

结合刚才针对满二叉树和普通二叉树的算法,上面这段代码应该不难理解,就是一个结合版,可是其中下降时间复杂度的技巧是很是微妙的

时间复杂度

开头说了,这个算法的时间复杂度是 O(logN*logN),这是怎么算出来的呢?

直觉感受好像最坏状况下是 O(N*logN) 吧,由于以前的 while 须要 logN 的时间,最后要 O(N) 的时间向左右子树递归:

return 1 + countNodes(root.left) + countNodes(root.right);
复制代码

关键点在于,这两个递归只有一个会真的递归下去,另外一个必定会触发hl == hr而当即返回,不会递归下去

为何呢?缘由以下:

一棵彻底二叉树的两棵子树,至少有一棵是满二叉树

image.png

看图就明显了吧,因为彻底二叉树的性质,其子树必定有一棵是满的,因此必定会触发hl == hr,只消耗 O(logN) 的复杂度而不会继续递归。

综上,算法的递归深度就是树的高度 O(logN),每次递归所花费的时间就是 while 循环,须要 O(logN),因此整体的时间复杂度是 O(logN*logN)。

因此说,「彻底二叉树」这个概念仍是有它存在的缘由的,不只适用于数组实现二叉堆,并且连计算节点总数这种看起来简单的操做都有高效的算法实现。

相关文章
相关标签/搜索