判断一棵树是否为完全二叉树

完全二叉树:若一棵二叉树具有具有n个节点,它的每个节点都与高度为k的满二叉树编号为0~n-1结点一一对应,则称这可二叉树为完全二叉树。

方法一:一维数组存储

根据完全二叉树的定义和性质,利用一位数组作为完全二叉树的存储,如下图

由图,节点的编号与数组元素的下标是一一对应的,可根据二叉树的性质,可方便找出下标
为i的的双亲结点a[i/2]及左右孩子结点a[i*2],a[i*2+1].这样判断一棵树是否为二叉树,应该对此二叉树从上到下,从左到右依次编号,
然后把编好的号依次存入一位数组中,在与相应深度的满二叉树的编号进行对比,即可判断此二叉树是否为完全二叉树。

但是该方法虽然实现容易,但占用空间太大,并且效率低,所以可通过层次遍历来判断是否为完全二叉树。

方法二:层次遍历(利用队列)

完全二叉树是指最后一层左边是满的,右边可能慢也不能不满,然后其余层都是满的,根据这个特性,利用层遍历。如果我们当前遍历到了NULL结点,如果后续还有非NULL结点,说明是非完全二叉树。

bool _CheckCompleteTree(Node *root)
    {
        queue<Node*> q;
        if (root == NULL)      //空树是完全二叉树
            return true;
        q.push(root);
        bool flag = false;
        while (!q.empty())
        {
            Node* front = q.front();
            if (front != NULL)
            {
                if (flag)
                    return false;
                q.push(front->_left);
                q.push(front->_right);
            }
            else
                flag = true;
            q.pop();
        }
        return true;
    }

完整代码及测试用例

#include<iostream>
#include<queue>
using namespace std;

template<class T>
struct BinaryTreeNode
{
    T _data;
    BinaryTreeNode *_left;
    BinaryTreeNode *_right;
    BinaryTreeNode(const T& d)
        :_data(d)
        , _left(NULL)
        , _right(NULL)
    {}
};

template<class T>
class BinaryTree
{
    typedef BinaryTreeNode<T> Node;
public:
    BinaryTree()
        :_root(NULL)
    {}
    BinaryTree(const T *a, size_t size, const T& invalid)
    {
        size_t index = 0;
        _root = _CreateNode(a, size, index, invalid);
    }
    void CheckCompleteTree()
    {
        bool ret;
        ret = _CheckCompleteTree(_root);
        cout << "Is a complate BinaryTree?:" << ret << endl;
    }

protected:
    bool _CheckCompleteTree(Node *root)
    {
        queue<Node*> q;
        if (root == NULL)      //空树是完全二叉树
            return true;
        q.push(root);
        bool flag = false;
        while (!q.empty())
        {
            Node* front = q.front();
            if (front != NULL)
            {
                if (flag)
                    return false;
                q.push(front->_left);
                q.push(front->_right);
            }
            else
                flag = true;
            q.pop();
        }
        return true;
    }
    Node * _CreateNode(const T* a, size_t size, size_t& index, const T& invalid)
    {
        Node* root = NULL;
        while ((index < size) && (a[index] != invalid))
        {
            root = new Node(a[index]);
            root->_left = _CreateNode(a, size, ++index, invalid);
            root->_right = _CreateNode(a, size, ++index, invalid);
        }
        return root;
    }
protected:
    Node* _root;
};
int main()
{
    int a[10] = { 1, 2, 3, ‘#‘, ‘#‘, 4, ‘#‘, ‘#‘, 5, 6, };
    BinaryTree<int> b1(a,10,‘#‘);
    b1.CheckCompleteTree();
    int a2[9] = { 1, 2, ‘#‘, 3,‘#‘, 4, ‘#‘, ‘#‘, 5 };
    BinaryTree<int> b2(a2, 9, ‘#‘);
    b2.CheckCompleteTree();

    system("pause");
    return 0;
}

时间: 2024-08-28 05:14:35

判断一棵树是否为完全二叉树的相关文章

1、如何判断一棵树是否是完全二叉树?

思路:通过树的层序遍历进行判断.结点入队时,当出现一个结点的孩子结点为空时,则之后就不能有新的结点入队. 若没有,则是完全二叉树,否则不是完全二叉树. 层序遍历代码: int after = 1;//表示还有结点入队 bool isComplete = true; void layerOrder(Node* root) { queue<Node*> Q; Q.push(root); while (!Q.empty()) { Node* front = Q.front(); Q.pop(); l

二叉树——判断一棵树是否是完全二叉树

二叉树按层遍历 判断条件:结点的左右孩子只有4种情况 其中的三种情况有特例 条件1.结点有右孩子,没有左孩子,直接返回false 条件2.结点左右孩子不全(有左没右,左右都没有),则后面遇到的所有结点,都必须是叶节点 只要不违反1.2的,就是完全二叉树 原文地址:https://www.cnblogs.com/SkyeAngel/p/8947531.html

判断一棵二叉树是否是完全二叉树

题目:判断一棵二叉树是否是完全二叉树 思路: 1.首先明确完全二叉树的概念.完全二叉树除最后一层外,所有层结点数均达到最大值,最后一层结点连续集中在最左边.空树也是完全二叉树. 2.我们可以通过层序遍历的方式遍历这个二叉树,使用一个队列存储遍历的结点.可以利用最后一层的结点集中在左侧这个特性解题,具体看代码: 代码: bool isCompleteTree(pTree pT) { Queue q; initQueue(&q); Enqueue(&q,pT);//根入队 pNode ptr

【LeetCode】Symmetric Tree 判断一棵树是否是镜像的

题目:Symmetric Tree <span style="font-size:18px;">/**LeetCode Symmetric Tree 对称的树 * 思路:判断一棵树是否对称,1.有左子树就要有右子树 * 2.除根节点外对称节点值要相同 * 注意:对称后就是左子树的左节点和右子树的右节点比较 * Definition for binary tree * public class TreeNode { * int val; * TreeNode left; *

判断一棵树是否是另一棵树的子树

问题 判断一棵树是否是另一棵树的子树,如图 思路 问题分两步: 找值相同的根结点(遍历解决) 判断两结点是否包含(递归:值.左孩子.右孩子分别相同) 代码 bool IsPart(TreeNode *root1, TreeNode *root2) { if (root2 == NULL) return true; if (root1 == NULL) return false; if (root1->val != root2->val) return false; return IsPart(

判断一棵树是否是二叉搜索树

前两天写过一篇博文<二叉搜索树基本操作实现>,为了更深入了解二叉搜索树的性质,本文实现判断一棵树是否为二叉搜索树算法. 二叉搜索树的性质: 任意节点的键值一定大于其左子树中的每一个节点的键值,并小于其右子树中的每一个节点的键值. 构造二叉树的节点定义为: struct TreeNode{ int data; TreeNode *left; TreeNode *right; }; 方法1 (错误) 对每一个节点,检测其值是否大于左子树节点,是否小于右子树节点.思路很简单,代码实现如下: bool

判断一棵树是否为二叉搜索树(二叉排序树) python

输入一棵树,判断这棵树是否为二叉搜索树.首先要知道什么是排序二叉树,二叉排序树是这样定义的,二叉排序树或者是一棵空树,或者是具有下列性质的二叉树: (1)若左子树不空,则左子树上所有结点的值均小于它的根结点的值: (2)若右子树不空,则右子树上所有结点的值均大于它的根结点的值: (3)左.右子树也分别为二叉排序树: (4)没有键值相等的节点 #方法1,直接判断 直接判断的关键在于不能只是单纯地判断根.左.右三个节点的大小关系,左子树的右节点不仅要大于父节点,还要小于父节点的父节点,右子树的左节点

判断一棵二叉树是否为BST,一棵树是否为完全二叉树

对于一颗搜索二叉树,最简单的方法就是用中序遍历,看是不是一个递增数列,如果是则是一颗搜索二叉树,如果不是则不是搜索二叉树.在这里用一个lastVisit去记录上一次搜索到的节点.整个过程就是先找到最左下角的节点,更新这个lastVisit为这个节点的值,然后按照中序遍历依次更新即可.代码如下. 1 #include <stdio.h> 2 #include <climits> 3 4 //二叉树结点 5 typedef struct binary_tree_node_t{ 6 bi

LeetCode&mdash;&mdash;Same Tree(判断两棵树是否相同)

问题: Given two binary trees, write a function to check if they are equal or not. Two binary trees are considered equal if they are structurally identical and the nodes have the same value.   分析: 考虑使用深度优先遍历的方法,同时遍历两棵树,遇到不等的就返回. 代码如下: /** * Definition f