java实现红黑树的添加

java实现红黑树(个人手写)

1.红黑树概念

平衡二叉查找树
(1)节点非红即黑
(2)根节点为黑色
(3)红色节点不能相邻
(4)任意节点到根节点黑色节点数相同
(5)叶子节点为黑色
在这里插入图片描述
红黑树展示网站

2.实现思路

红黑树什么时候发生旋转?
父节点为红色 叔叔节点为null或者为黑色
个人认为分为下面四种情况:
(1)0001 需要旋转变成 0002 的 左孩子
在这里插入图片描述
(2)0003 需要旋转成 0002 的右孩子
在这里插入图片描述
(3) 0003 需要旋转成 0002 的左孩子, 然后就变成了情况(1)
在这里插入图片描述
(4)0001 需要旋转成 0002 的右孩子,然后就变成了情况(2)
在这里插入图片描述
红黑树的变色
对于上面四种情况,因为插入的节点必是红色,所以父亲要变成黑色,祖父要变成红色,如果说祖父是根节点,那么就不用变色

3.代码实现

(1)定义树的节点

public class Node {
	// 节点的值
    int value;
    // 右节点
    Node right;
    // 左节点
    Node left;
    // 颜色 为了我看着不眼花,就直接 black red
    String color;
    // 父亲节点
    Node parent;
    
    // 构造方法
	public Node(int value) {
        this.value = value;
    }
	
    @Override
    public String toString() {
        return "Node{" +
                "value=" + value +
                ", right=" + right +
                ", left=" + left +
                ", color=" + color +
                '}';
    }
}

(2)代码实现

public class RedBlackTree {
	// 根节点
    Node root;

	// 这两个参数是想知道 pDir父亲是祖父的左0还是右1  dir自己是父亲的左0还是右1
    private int dir = 0;
    private int pDir = 0;

    /**
     * 添加
     */
    public void add(int val) {
        Node node = new Node(val);
        // 如果根节点为空,那么为第一个元素,节点为黑色
        if (root == null) {
            node.color = "black";
            root = node;
        } else {
            // 不为空 节点为红色
            node.color = "red";
            insertNode(root, node);
        }
        repairTree(node);
    }

    /**
     * 插入节点
     */
    public void insertNode(Node compare, Node add) {
        // 大于往右走 小于往左走
        if (compare.value < add.value) {
            // 右子节点为空则将添加节点设为右节点 不为空则与右节点进行比较
            if (compare.right == null) {
                compare.right = add;
                dir = 1;
                add.parent = compare;
            } else {
                insertNode(compare.right, add);
            }
        } else if (compare.value > add.value) {
            // 左子节点为空则将添加节点设为左节点 不为空则与左节点进行比较
            if (compare.left == null) {
                compare.left = add;
                dir = 0;
                add.parent = compare;
            } else {
                insertNode(compare.left, add);
            }
        }
    }

    /**
     * 修复树
     *
     * @param node
     */
    private void repairTree(Node node) {
    	// 父亲
        Node parent = node.parent;
        // 祖父
        Node grandParent = null;
        if (parent != null) {
            grandParent = node.parent.parent;
        }
        // 叔叔
        Node uncle = null;
        if (grandParent != null) {
            // 确认父亲是祖父的左孩子还是右孩子
            if (grandParent.right != null && grandParent.right == parent) {
                // 父亲是祖父的右孩子
                uncle = grandParent.left;
                pDir = 1;
            } else if (grandParent.left != null && grandParent.left == parent) {
                // 父亲是祖父的左孩子
                uncle = grandParent.right;
                pDir = 0;
            }
        }

        // 父亲为红色
        if (parent != null && "red".equals(parent.color)) {
            if (uncle == null || "black".equals(uncle.color)) {
                // 叔叔为空 或者为黑色 旋转 变色
                if (pDir == 1 && dir == 1) {
                    leftRevolve(grandParent, parent);
                    parent.color = "black";
                    grandParent.color = "red";
                } else if (pDir == 1 && dir == 0) {
                    rightRevolve(parent, node);
                    leftRevolve(grandParent, node);
                    node.color = "black";
                    grandParent.color = "red";
                } else if (pDir == 0 && dir == 1) {
                    leftRevolve(parent, node);
                    rightRevolve(grandParent, node);
                    node.color = "black";
                    grandParent.color = "red";
                } else if (pDir == 0 && dir == 0) {
                    rightRevolve(grandParent, parent);
                    parent.color = "black";
                    grandParent.color = "red";
                }
            } else if ("red".equals(uncle.color)) {
                // 叔叔为红色 变色
                uncle.color = "black";
                parent.color = "black";
                if (grandParent != root) {
                    grandParent.color = "red";
                }
            }
        }
    }

	
    public void leftRevolve(Node a, Node b) {
        a.right = null;
        b.parent = a.parent;
        if (a.parent == null) {
            root = b;
        } else {
            if (a.parent.right == a) {
                a.parent.right = b;
            } else if (a.parent.left == a) {
                a.parent.left = b;
            }
        }
        a.parent = b;
        a.right = b.left;
        b.left = a;

    }

    public void rightRevolve(Node a, Node b) {
        a.left = null;
        b.parent = a.parent;
        if (a.parent == null) {
            root = b;
        } else {
            if (a.parent.right == a) {
                a.parent.right = b;
            } else if (a.parent.left == a) {
                a.parent.left = b;
            }
        }
        a.parent = b;
        a.left = b.right;
        b.right = a;
    }
}

个人看法,感谢观看!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值