7 Star 32 Fork 21

空無一悟 / algorithms

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
BSTKV_RBTree.java 17.46 KB
一键复制 编辑 原始数据 按行查看 历史
空無一悟 提交于 2021-09-22 23:47 . init
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583
/***********************************************************
* @Description : 基于支持键值对的二分搜索树实现红黑树
* 13.3 实现红黑树的基础结构
* 13.5 实现红黑树的元素添加
* @author : 梁山广(Liang Shan Guang)
* @date : 2020/1/5 10:56
* @email : liangshanguang2@gmail.com
***********************************************************/
package Chapter13ReadBlackTree.Section5to7AddNode;
import java.util.ArrayDeque;
import java.util.Queue;
/**
* @param <K> Key表示键值对的键,继承Comparable使得二叉树的元素是可比较的,可以用compareTo()方法进行比较。
* @param <V> 即Val,是键值对的值即value
* @author liangshanguang
*/
public class BSTKV_RBTree<K extends Comparable<K>, V> {
/**
* 红黑树节点颜色的静态变量
*/
private static final boolean RED = true, BLACK = false;
/**
* 二分搜索树每个节点的封装类。Node是BST的内部类,所以属性可以设置为public
*/
private class Node {
/**
* 当前节点的键
*/
public K key;
public V val;
/**
* 节点e的左子节点和右子节点
*/
public Node left, right;
/**
* 节点颜色,因为只有红黑两种可能,所以用布尔变量表示。我们约定true为红色,false为黑色
*/
private boolean color;
public Node(K key, V val) {
this.key = key;
this.val = val;
left = null;
right = null;
// 默认每个点的初始颜色是红色(true),因为我们新加入的节点一般都是和现有的节点做成3节点或4节点,
// 当表示成红黑树节点时,新加入的节点往往都是在红色的位置上,初始化成红色省心
color = RED;
}
@Override
public String toString() {
return "Node{" +
"key=" + key +
", val=" + val +
'}';
}
}
/**
* 二分搜索树的根节点
*/
private Node root;
/**
* 二分搜索树的节点个数
*/
private int size;
public BSTKV_RBTree() {
root = null;
size = 0;
}
/**
* 获取二分搜索树的节点总数
*/
public int getSize() {
return size;
}
/**
* 判断二分搜索树是否为空
*
* @return 通过节点总数是否为0来进行判断
*/
public boolean isEmpty() {
return size == 0;
}
/**
* 判断节点node是否为红色,考虑空的情况
*/
private boolean isRed(Node node) {
if (node == null) {
// 空节点被认为是黑节点
return BLACK;
}
return node.color;
}
/**
* 新加入节点后进行左旋转
* node.right = T2;
* x.left=node;
* x.color = node.color;
* node.color = RED
* 图示如下:
* // node x
* // / \ 左旋转 / \
* // T1 x ---------> node T3
* // / \ / \
* // T2 T3 T1 T2
*
* @param node 新加入节点的父节点
* @return 左旋转后原本以node作为根节点的子树的新的根节点
*/
private Node rotateLeft(Node node) {
// 暂存节点
Node x = node.right;
Node T2 = x.left;
// 左旋转
node.right = T2;
x.left = node;
// 更新颜色
x.color = node.color;
node.color = RED;
return x;
}
/**
* 新加入节点后进行右旋转
* 下面的伪代码
* node.left = T1
* x.right = node
* x.color = node.color
* node.color = RED
* flipColors()
* 下面是图示:
* // node x
* // / \ 右旋转 / \
* // x T2 -------> y node
* // / \ / \
* // y T1 T1 T2
*
* @param node 新加入节点的父节点
* @return 右旋转后原本以node作为根节点的子树的新的根节点
*/
private Node rotateRight(Node node) {
// 暂存节点
Node x = node.left;
Node T1 = x.right;
// 右旋转
node.left = T1;
x.right = node;
// 颜色更新
x.color = node.color;
node.color = RED;
return x;
}
/**
* 把以node为根的节点和左右孩子节点的颜色进行翻转(红变黑,黑变红)
* 红黑树中新插入的节点和已有节点组成了4节点,而且是平衡的一个小树,则需要进行颜色翻转.
*
* @param node 要翻转子树的根节点
*/
private void flipColors(Node node) {
node.color = RED;
node.left.color = BLACK;
node.right.color = BLACK;
}
/**
* 红黑树的颜色和节点平衡
*
* @param node 要维护的节点
* @return 维护后新的根节点
*/
private Node rbManage(Node node) {
// 左孩子不是红色,右孩子是红色,就左旋转
if (!isRed(node.left) && isRed(node.right)) {
node = rotateLeft(node);
}
// 左孩子是红色,左孩子的左孩子还是红色,就左旋转
if (isRed(node.left) && isRed(node.left.left)) {
node = rotateRight(node);
}
// 左孩子和右孩子都是红色
if (isRed(node.left) && isRed(node.right)) {
flipColors(node);
}
return node;
}
/**
* 向以节点Node为根节点的红黑树中添加新的键值对元素,递归实现
*
* @param node 红黑树的根节点
* @param key 要加入地节点的键
* @param val 要加入地节点的值
* @return 当前子树添加节点后的红黑树的根节点
*/
private Node add(Node node, K key, V val) {
// 递归终止条件
if (node == null) {
// 只要碰到了为空的node,就一定要把我们的e作为节点添加到这里的,具体是作为左子树、右子树还是根节点到下面再进行设置
size++;
return new Node(key, val);
}
// 递归组成逻辑
if (key.compareTo(node.key) < 0) {
// key小于根节点的key,往node的左子树继续遍历
node.left = add(node.left, key, val);
} else if (key.compareTo(node.key) > 0) {
// key大于根节点的key,往node的右子树继续遍历
node.right = add(node.right, key, val);
} else {
// 如果和遍历到的节点相等即key.compareTo(node.key)==0,则进行节点值更新
node.val = val;
}
// 当这个node是把key给new出来地就设置到子节点为空的上面去;如果不是new出来地相当于把已有的二分搜索树中的节点关系又设置一次
return rbManage(node);
}
/**
* 向红黑树中添加键值对节点(key, val)
*/
public void add(K key, V val) {
root = add(root, key, val);
// 保持根节点为黑色
root.color = BLACK;
}
/**
* 在以节点node为根节点的二分搜索树树中查找是否包含键key的键值对(key, val)节点,有地话就把这个节点返回
*/
private Node getNode(Node node, K key) {
// 递归终止条件
if (node == null) {
// 遍历到到二叉树最底部了,还没找到,二分搜索树中肯定没有这个这个元素了
return null;
}
// 递归组成逻辑
if (key.compareTo(node.key) < 0) {
// key小于当前节点的key,向左递归
return getNode(node.left, key);
} else if (key.compareTo(node.key) > 0) {
// key大于当前节点的key,向右递归
return getNode(node.right, key);
} else {
// key等于当前节点的key,返回当前节点node
return node;
}
}
/**
* 看二分搜索树种是否包含键为key的键值对
*/
public boolean contains(K key) {
// 找不到指定的节点就返回null
return getNode(root, key) != null;
}
/**
* 根据指定的键查找对应的值
*
* @param key 键
* @return 值
*/
public V get(K key) {
Node node = getNode(root, key);
return node == null ? null : node.val;
}
/**
* 更新树种的键值对
*
* @param key 键
* @param valNew 值
*/
public void set(K key, V valNew) {
Node node = getNode(root, key);
if (node == null) {
// 原来不存在这个节点就加入
add(key, valNew);
} else {
node.val = valNew;
}
}
/**
* 前序遍历以node作为根节点的二分搜索树
*/
private void preOrder(Node node) {
// 递归终止条件
if (node == null) {
// 遍历到null节点就返回上一层递归
return;
}
// 递归组成逻辑
// 1.访问当前节点。需要存储时可以放到list中
System.out.print(node.key + ":" + node.val + " ");
// 2.遍历左子树
preOrder(node.left);
// 3.遍历右子树
preOrder(node.right);
}
/**
* 前序遍历
*/
public void preOrder() {
preOrder(root);
}
/**
* 中序遍历以node作为根节点的二分搜索树
*/
private void inOrder(Node node) {
// 递归终止条件
if (node == null) {
// 遍历到null节点就返回上一层递归
return;
}
// 递归组成逻辑
// 2.遍历左子树
inOrder(node.left);
// 1.访问当前节点。需要存储时可以放到list中
System.out.print(node.key + ":" + node.val + " ");
// 3.遍历右子树
inOrder(node.right);
}
/**
* 中序遍历
*/
public void inOrder() {
inOrder(root);
}
/**
* 后序遍历以node作为根节点的二分搜索树
*/
private void postOrder(Node node) {
// 递归终止条件
if (node == null) {
// 遍历到null节点就返回上一层递归
return;
}
// 递归组成逻辑
// 2.遍历左子树
postOrder(node.left);
// 3.遍历右子树
postOrder(node.right);
// 1.访问当前节点。需要存储时可以放到list中
System.out.print(node.key + ":" + node.val + " ");
}
/**
* 后序遍历
*/
public void postOrder() {
postOrder(root);
}
/**
* 层序遍历
*/
public void levelOrder() {
levelOrder(root);
}
/**
* 层序遍历以node作为根节点的二分搜索树
*
* @param root 二分搜索树的根节点
*/
private void levelOrder(Node root) {
Queue<Node> queue = new ArrayDeque<>();
queue.add(root);
while (!queue.isEmpty()) {
Node node = queue.remove();
// 每次弹出一个元素就打印下
System.out.print(node.key + ":" + node.val + " ");
if (node.left != null) {
queue.add(node.left);
}
if (node.right != null) {
queue.add(node.right);
}
}
}
/**
* 寻找以node作为跟节点的二分搜索树的最小节点
*
* @param node 根节点
*/
private Node minimum(Node node) {
if (node.left == null) {
return node;
}
return minimum(node.left);
}
/**
* 寻找二分搜索树的最小元素
*/
public Node minimum() {
if (size == 0) {
throw new IllegalArgumentException("BST为空!无法找最小值");
}
return minimum(root);
}
/**
* 寻找以node作为跟节点的二分搜索树的最大节点
*
* @param node 根节点
*/
private Node maximum(Node node) {
if (node.right == null) {
return node;
}
return maximum(node.right);
}
/**
* 寻找二分搜索树的最大元素
*/
public Node maximum() {
if (size == 0) {
throw new IllegalArgumentException("BST为空!无法找最小值");
}
return maximum(root);
}
/**
* 从二分搜索树种删除最小元素并返回
*
* @return 删除的最小元素
*/
public Node removeMin() {
// 这里先记录下最小值,防止下面removeMin()后树结构变了
Node min = minimum();
// 这个递归函数返回地是新子树的根节点,不是我们要找地最小值
root = removeMin(root);
// 直接用我们前面找到的最小元素来返回
return min;
}
/**
* 删除以node作为根节点的二分搜索树中的最小节点,并返回删除节点后的新的二分搜索树的根节点
*
* @param node 根节点
*/
private Node removeMin(Node node) {
// 递归终止条件
if (node.left == null) {
// 递归遍历到左子树为空,说明找到了最小节点node
// node.right是否为空都可以正常返回给上一级的父节点来设置父节点的左节点直接指向当前节点的右子树
Node rightNode = node.right;
node.right = null;
size--;
return rightNode;
}
// 递归组成逻辑
// 当左节点不是null时就正常往下递归,返回当前节点给上一层节点设置下自己的左节点
node.left = removeMin(node.left);
return node;
}
/**
* 从二分搜索树种删除最大元素并返回
*
* @return 删除的最大元素
*/
public Node removeMax() {
// 这里先记录下最大值,removeMax()后树结构变了
Node max = maximum();
// 这个递归函数返回地是新子树的根节点,不是我们要找地最大值
root = removeMax(root);
// 直接用我们前面找到的最大元素来返回
return max;
}
/**
* 删除以node作为根节点的二分搜索树中的最大节点,并返回删除节点后的新的二分搜索树的根节点
*
* @param node 根节点
*/
private Node removeMax(Node node) {
// 递归终止条件
if (node.right == null) {
// 递归遍历到右子树为空,说明找到了最大节点node
// node.left是否为空都可以正常返回给上一级的父节点来设置父节点的右节点直接指向当前节点的左子树
Node leftNode = node.left;
node.left = null;
size--;
return leftNode;
}
// 递归组成逻辑
// 当右节点不是null时就正常往下递归,返回当前节点给上一层节点设置下自己的右节点
node.right = removeMax(node.right);
return node;
}
/**
* 删除指定值的结点
*
* @param key 节点的键
*/
public void remove(K key) {
root = remove(root, key);
}
/**
* 删除
*
* @param node 二分搜索树的根节点
* @param key 待删除节点的键
* @return 要挂载到当前节点父节点的子树
*/
private Node remove(Node node, K key) {
// 递归终止条件
if (node == null) {
return null;
}
// 递归组成逻辑
// 还没找到就接着往下找
if (key.compareTo(node.key) < 0) {
// 要找的值比当前节点小,向左递归
node.left = remove(node.left, key);
return node;
} else if (key.compareTo(node.key) > 0) {
// 要找的值比当前节点大,向右递归
node.right = remove(node.right, key);
return node;
} else {
// node.key == key 找到相等的节点了,下面删除指定值的节点
if (node.left == null) {
Node rightNode = node.right;
// 释放引用
node.right = null;
size--;
// 左节点为空,把node的右子树挂接到node的父亲节点即可
return rightNode;
}
if (node.right == null) {
Node leftNode = node.left;
// 释放引用
node.left = null;
size--;
// 右节点为空,把node的左子树挂接到node的父亲节点即可
return leftNode;
}
// node的左右子树都不为空,就找node的右子树的最小值来代替node
Node minimumRight = minimum(node.right);
// 警告:下面两行代码一定不要颠倒,一定要先设置right再设置left,否则会出现迭代引用!因为下面那行改变了node.right的结构。参考问题:https://coding.imooc.com/learn/questiondetail/143936.html
// 选出node右子树最小元素来代替node,那么右子树最小元素就要从原来位置删掉
minimumRight.right = removeMin(node.right);
// 替换当前节点node的左右子树
minimumRight.left = node.left;
// 释放node的引用
node.left = node.right = null;
// 返回给上一级来设置父节点
return minimumRight;
}
}
}
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
1
https://gitee.com/lsgwr/algorithms.git
git@gitee.com:lsgwr/algorithms.git
lsgwr
algorithms
algorithms
master

搜索帮助