PAT_甲级_1123 Is It a Complete AVL Tree

题目大意

给定一颗平衡二叉树的插入序列,要求输出它的层序序列并判断是否为完全二叉树。

算法思路

首先需要对该AVL树进行建树操作(具体不细说,看代码即可),然后对该树进行层序遍历,一边输出层序变量序列,一遍判断该树是否是完全二叉树
判断二叉树是否是完全二叉树的方法为:

  • 1、如果当前节点只有右孩子,一定不是。
  • 2、如果当前节点有孩子,但是其前驱节点中存在孩子缺失的情况,那么一定不是。

提交结果

PAT_甲级_1123 Is It a Complete AVL Tree_第1张图片

AC代码

#include
#include
#include

using namespace std;

struct Node {
    int v;
    Node *left;
    Node *right;
    int height;
};

int n;

//生成新的结点
Node *newNode(int v) {
    Node *w = new Node;
    w->v = v;
    w->height = 1;
    w->left = nullptr;
    w->right = nullptr;
    return w;
}

int getHeight(Node *root) {
    if (root == nullptr) {
        return 0;
    } else {
        return root->height;
    }
}

int getBalancedFactor(Node *root) {
    return getHeight(root->left) - getHeight(root->right);
}

void updateHeight(Node *&root) {
    root->height = max(getHeight(root->left), getHeight(root->right)) + 1;
}

void L(Node *&root) {
    Node *temp = root->right;
    root->right = temp->left;
    temp->left = root;
    updateHeight(root);
    updateHeight(temp);
    root = temp;
}

void R(Node *&root) {
    Node *temp = root->left;
    root->left = temp->right;
    temp->right = root;
    updateHeight(root);
    updateHeight(temp);
    root = temp;
}

void insert(Node *&root, int x) {
    if (root == nullptr) {
        root = newNode(x);
        return;
    }
    if (root->v < x) {
        insert(root->right, x);
        // 插入完成后得得更新root的高度
        updateHeight(root);
        if (getBalancedFactor(root) == -2) {
            if (getBalancedFactor(root->right) == -1) {
                //RR型需要对root左旋一次
                L(root);
            } else if (getBalancedFactor(root->right) == 1) {
                // RL型,先对root->right右旋,再对root左旋
                R(root->right);
                L(root);
            }
        }
    } else {
        insert(root->left, x);
        // 插入完成后得得更新root的高度
        updateHeight(root);
        if (getBalancedFactor(root) == 2) {
            if (getBalancedFactor(root->left) == 1) {
                // LL型需要对root右旋一次
                R(root);
            } else if (getBalancedFactor(root->left) == -1) {
                // LR型,先对root->left左旋,再对root右旋
                L(root->left);
                R(root);
            }
        }
    }
}

Node *createTree(int data[]) {
    Node *root = nullptr;
    for (int i = 0; i < n; ++i) {
        insert(root, data[i]);
    }
    return root;
}

bool isComplete = true;
bool onlyLeft = false;// 记录前驱是否只有左孩子
bool noChildren = false;// 记录前驱是否没有孩子

int num = 0;// 打印控制
void layerOrder(Node *root) {
    queue q;
    q.push(root);
    bool flag1 = false, flag2 = false;
    while (!q.empty()) {
        Node *t = q.front();
        q.pop();
        printf("%d", t->v);
        if (num < n - 1) {
            printf(" ");
        } else {
            printf("\n");
        }
        ++num;
        if (t->left) {
            q.push(t->left);
        }
        if (t->right) {
            q.push(t->right);
        }
        // 判断是否是完全二叉树
        if (t->left == nullptr && t->right != nullptr) {
            // 只有右孩子
            isComplete = false;
        } else if(noChildren||onlyLeft){
            if(t->left != nullptr||t->right != nullptr){
                // 前驱出现只有左孩子或者没有孩子的情况,当前节点还有孩子
                isComplete = false;
            }
        } else if (t->left == nullptr && t->right == nullptr){
            // 没有孩子
            noChildren = true;
        } else if(t->left != nullptr && t->right == nullptr){
            // 只有左孩子
            onlyLeft = true;
        }
    }
}

int main() {
    scanf("%d", &n);
    int data[n];
    for (int i = 0; i < n; ++i) {
        scanf("%d", &data[i]);
    }
    Node *root = createTree(data);
    layerOrder(root);
    if (isComplete) {
        printf("YES");
    } else {
        printf("NO");
    }
    return 0;
}

你可能感兴趣的:(c++数据结构和算法)