基于紅黑樹插入操作原理及java實(shí)現(xiàn)方法(分享)
紅黑樹是一種二叉平衡查找樹,每個(gè)結(jié)點(diǎn)上有一個(gè)存儲(chǔ)位來表示結(jié)點(diǎn)的顏色,可以是RED或BLACK。
紅黑樹具有以下性質(zhì):
(1) 每個(gè)結(jié)點(diǎn)是紅色或是黑色
(2) 根結(jié)點(diǎn)是黑色的
(3) 如果一個(gè)結(jié)點(diǎn)是紅色的,則它的兩個(gè)兒子都是黑色的
(4) 對(duì)于每個(gè)結(jié)點(diǎn),從該結(jié)點(diǎn)到其子孫結(jié)點(diǎn)的所有路徑上包含相同數(shù)目的黑結(jié)點(diǎn)
通過紅黑樹的性質(zhì),可以保證所有基于紅黑樹的實(shí)現(xiàn)都能保證操作的運(yùn)行時(shí)間為對(duì)數(shù)級(jí)別(范圍查找除外。它所需的額外時(shí)間和返回的鍵的數(shù)量成正比)。
Java的TreeMap就是通過紅黑樹實(shí)現(xiàn)的。
紅黑樹的操作如果不畫圖很容易搞糊涂,下面通過圖示來說明紅黑樹的插入操作。
插入一個(gè)紅色的節(jié)點(diǎn)到紅黑樹中之后,會(huì)有6種情況:圖示中N表示插入的節(jié)點(diǎn),P表示父節(jié)點(diǎn),U表示叔叔節(jié)點(diǎn),G表示祖父節(jié)點(diǎn),X表示當(dāng)前操作節(jié)點(diǎn)




代碼如下:
public class RedBlackBST<Key extends Comparable<Key>, Value> {
private Node root;
private static final boolean RED = true;
private static final boolean BLACK = false;
private class Node{
private Key key; //鍵
private Value val; //值
private Node left, right, parent; //左右子樹和父節(jié)點(diǎn)
private boolean color; //由其父節(jié)點(diǎn)指向它的鏈接的顏色
public Node(Key key, Value val,Node parent, boolean color){
this.key = key;
this.val = val;
this.color = color;
}
}
public Value get(Key key){
Node x = root;
while(x!=null){
int cmp = key.compareTo(x.key);
if(cmp < 0 ) x = x.left;
else if(cmp > 0) x = x.right;
else return x.val;
}
return null;
}
public void put(Key key, Value val){
if(root==null) { //如果是根節(jié)點(diǎn),就將節(jié)點(diǎn)新建為黑色
root = new Node(key,val,null,BLACK);
return;
}
//尋找合適的插入位置
Node parent = null;
Node cur = root;
while(cur!=null) {
parent = cur;
if(key.compareTo(cur.key)>0) cur=cur.right;
else cur = cur.left;
}
Node n = new Node(key,val,parent,RED); //普通的新建節(jié)點(diǎn)為紅色
//將新節(jié)點(diǎn)插入parent下
if(key.compareTo(parent.key) > 0) parent.right = n;
else parent.left = n;
//插入新節(jié)點(diǎn)后要調(diào)整樹中部分節(jié)點(diǎn)的顏色和屬性來保證紅黑樹的特征不被破壞
fixAfterInsertion(n);
}
private Node parentOf(Node x) {
return (x==null ? null : x.parent);
}
private boolean colorOf(Node x) {
return (x==null ? BLACK : x.color);
}
private Node leftOf(Node x) {
return (x==null ? null : x.left);
}
private Node rightOf(Node x) {
return(x==null ? null : x.right);
}
private void setColor(Node x, boolean color) {
if(x!=null)
x.color = color;
}
private void fixAfterInsertion(Node x) {
while(x!=null && colorOf(parentOf(x)) == RED) {
Node grandPa = parentOf(parentOf(x));
Node parent = parentOf(x);
if(parent == leftOf(grandPa)) {//case 1 || case2 || case3
Node uncle = rightOf(grandPa);
if(colorOf(uncle) == RED) {//case1, uncle is red
setColor(parent,BLACK); //父節(jié)點(diǎn)置黑
setColor(uncle, BLACK); //叔叔節(jié)點(diǎn)置黑
setColor(grandPa,RED); //祖父節(jié)點(diǎn)置紅
x = grandPa; //因?yàn)樽娓腹?jié)點(diǎn)由黑轉(zhuǎn)紅,故要重新調(diào)整父節(jié)點(diǎn)及其祖先的紅黑屬性
}else {//case2 || case3,uncle is black
if(x==rightOf(parent)) { //case2
x = parent;
rotateLeft(x);
}
//case3
setColor(parent,BLACK);
setColor(grandPa, RED);
rotateRight(grandPa);
}
}else {//case4 || case 5 || case6
Node uncle = leftOf(grandPa);
if(colorOf(uncle) == RED) { //case4 || case5 || case6
setColor(parent,BLACK);
setColor(uncle, BLACK);
setColor(grandPa,RED);
x = grandPa;
}else{ //case5 || case6, uncle is black
if(x==leftOf(parent)) { //case5
x = parent;
rotateRight(x);
}
//case6
setColor(parent,BLACK);
setColor(grandPa, RED);
rotateLeft(grandPa);
}
}
}
}
private void rotateLeft(Node x) {
if(x==null) return;
Node y = x.right;
x.right = y.left;
if(y.left!=null)
y.left.parent = x;
y.left = x;
y.parent = x.parent;
if(x.parent == null) {
root = y;
}
else if(x.parent.left == x) {
x.parent.left = y;
}else {
x.parent.right = y;
}
x.parent = y;
}
private void rotateRight(Node x) {
if(x==null) return;
Node y = x.left;
x.left = y.right;
if(y.right != null)
y.right.parent = x;
y.right = x;
y.parent = x.parent;
if(x.parent == null) {
root = y;
}else if(x.parent.left==x) {
x.parent.left = y;
}else {
x.parent.right=y;
}
x.parent = y;
}
}
上面的rotateLeft和rotateRight有必要畫個(gè)圖示:

以上這篇基于紅黑樹插入操作原理及java實(shí)現(xiàn)方法(分享)就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
java實(shí)現(xiàn)短地址服務(wù)的方法(附代碼)
大多數(shù)情況下URL太長(zhǎng),字符多,不便于發(fā)布復(fù)制和存儲(chǔ),本文就介紹了通過java實(shí)現(xiàn)短地址服務(wù),減少了許多使用太長(zhǎng)URL帶來的不便,需要的朋友可以參考下2015-07-07
springboot json時(shí)間格式化處理的方法
這篇文章主要介紹了springboot json時(shí)間格式化處理的方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-03-03
Java使用Apache Commons高效處理CSV文件的操作指南
在 Java 開發(fā)中,CSV(Comma-Separated Values,逗號(hào)分隔值)是一種常見的數(shù)據(jù)存儲(chǔ)格式,廣泛用于數(shù)據(jù)交換和簡(jiǎn)單的存儲(chǔ)任務(wù),本文將介紹Java使用Apache Commons高效處理CSV文件的操作指南,需要的朋友可以參考下2025-03-03

