数据结构:简化平衡二叉树旋转时节点交换的操作

数据结构 About 5,103 words

说明

之前文章二叉树左旋、右旋、双旋时替换是new一个临时节点,使用根节点指向不变替换值的方式完成。可参见:https://www.zhangbj.com/p/734.html

此处使用接收返回值,令根节点重新指向旋转后的树。

左旋

public Node leftRotate(Node n) {
    Node x = n.right;
    n.right = x.left;
    x.left = n;
    return x;
}

右旋

public Node rightRotate(Node n) {
    Node x = n.left;
    n.left = x.right;
    x.right = n;
    return x;
}

添加节点

判断左右旋添加还是一样。高度差大于1、左右节点高度不相等都需做出旋转。

public Node add(Node rootNode, Node n) {
    if (rootNode == null) {
        return n;
    }
    if (rootNode.id > n.id) { // 放左边
        rootNode.left = add(rootNode.left, n);
    } else { // 放右边
        rootNode.right = add(rootNode.right, n);
    }

    if (rootNode.rightHeight() - rootNode.leftHeight() > 1) {
        if (rootNode.right != null && rootNode.right.leftHeight() > rootNode.right.rightHeight()) {
            rootNode.right = rightRotate(rootNode.right);
        }
        rootNode = leftRotate(rootNode);
    } else if (rootNode.leftHeight() - rootNode.rightHeight() > 1) {
        if (rootNode.left != null && rootNode.left.rightHeight() > rootNode.left.leftHeight()) {
            rootNode.left = leftRotate(rootNode.left);
        }
        rootNode = rightRotate(rootNode);
    }
    return rootNode;
}

完整代码 - Java 实现

public class AVLTree {

    public static void main(String[] args) {
        AVLTree tree = new AVLTree();
//        int[] arr = {4, 3, 6, 5, 7, 8}; // 左旋
//        int[] arr = {10, 12, 8, 9, 7, 6}; // 右旋
        int[] arr = {10, 11, 7, 6, 8, 9}; // 双旋
        for (int i : arr) {
            Node node = new Node(i);
            tree.add(node);
        }

        tree.printNode();

        System.out.println("-------------");

        System.out.println("tree height#" + tree.height());

        System.out.println("tree root#" + tree.root.id);
        System.out.println("tree root left#" + tree.root.left.id);
        System.out.println("tree root right#" + tree.root.right.id);
        System.out.println("tree leftHeight#" + tree.root.leftHeight());
        System.out.println("tree rightHeight#" + tree.root.rightHeight());

    }

    public Node root;

    public void add(Node n) {
        root = add(root, n);
    }

    public Node add(Node rootNode, Node n) {
        if (rootNode == null) {
            return n;
        }
        if (rootNode.id > n.id) { // 放左边
            rootNode.left = add(rootNode.left, n);
        } else { // 放右边
            rootNode.right = add(rootNode.right, n);
        }

        if (rootNode.rightHeight() - rootNode.leftHeight() > 1) {
            if (rootNode.right != null && rootNode.right.leftHeight() > rootNode.right.rightHeight()) {
                rootNode.right = rightRotate(rootNode.right);
            }
            rootNode = leftRotate(rootNode);
        } else if (rootNode.leftHeight() - rootNode.rightHeight() > 1) {
            if (rootNode.left != null && rootNode.left.rightHeight() > rootNode.left.leftHeight()) {
                rootNode.left = leftRotate(rootNode.left);
            }
            rootNode = rightRotate(rootNode);
        }
        return rootNode;
    }

    public Node leftRotate(Node n) {
        Node x = n.right;
        n.right = x.left;
        x.left = n;
        return x;
    }

    public Node rightRotate(Node n) {
        Node x = n.left;
        n.left = x.right;
        x.right = n;
        return x;
    }

    public int height() {
        return this.root.height();
    }

    public void printNode() {
        root.infixOrder();
    }

    private static class Node {
        public int id;
        public Node left;
        public Node right;

        public Node(int id) {
            this.id = id;
        }

        public int height() {
            int leftHeight;
            if (this.left == null) {
                leftHeight = 0;
            } else {
                leftHeight = this.left.height();
            }


            int rightHeight;
            if (this.right == null) {
                rightHeight = 0;
            } else {
                rightHeight = this.right.height();
            }
            return Math.max(leftHeight + 1, rightHeight + 1);
        }

        public int leftHeight() {
            if (this.left == null) {
                return 0;
            } else {
                return this.left.height();
            }
        }

        public int rightHeight() {
            if (this.right == null) {
                return 0;
            } else {
                return this.right.height();
            }
        }

        public void infixOrder() {
            if (this.left != null) {
                this.left.infixOrder();
            }
            System.out.println("node id#" + this.id);
            if (this.right != null) {
                this.right.infixOrder();
            }
        }

    }

}

输出:

node id#6
node id#7
node id#8
node id#9
node id#10
node id#11
-------------
tree height#3
tree root#8
tree root left#7
tree root right#10
tree leftHeight#2
tree rightHeight#2
Views: 1,916 · Posted: 2021-02-20

————        END        ————

Give me a Star, Thanks:)

https://github.com/fendoudebb/LiteNote

扫描下方二维码关注公众号和小程序↓↓↓

扫描下方二维码关注公众号和小程序↓↓↓


Today On History
Browsing Refresh