6.樹(二,二叉排序樹)

image
import javax.swing.tree.TreeNode;
import java.util.NoSuchElementException;

/**
 * 二分查找樹
 */
public class SearchBinaryTree<T extends Comparable> {

    TreeNode root;//根結(jié)點(diǎn)

    public SearchBinaryTree() {
    }

    /**
     * 添加一個(gè)結(jié)點(diǎn)
     * <p>
     * 首先通過比較大小找到需要添加的結(jié)點(diǎn)的父節(jié)點(diǎn)parent,
     * 然后根據(jù)結(jié)點(diǎn)值與parent值大小,將該結(jié)點(diǎn)添加到parent的leftChild或者rightChild
     *
     * @param data 輸入要添加的值
     * @return 返回添加的結(jié)點(diǎn)本身
     */
    public TreeNode put(T data) {
        if (root == null) {
            TreeNode node = new TreeNode(data);
            root = node;
            return node;
        } else {
            TreeNode parent = null;
            TreeNode targetNode = root;
            while (targetNode != null) {
                parent = targetNode;
                if (data.compareTo(targetNode.data) > 0) {
                    targetNode = targetNode.rightChild;
                } else if (data.compareTo(targetNode.data) < 0) {
                    targetNode = targetNode.leftChild;
                } else {
                    //SearchBinaryTree contains this data,do not need to put
                    return targetNode;
                }
            }

            TreeNode node = new TreeNode(data);
            node.parent = parent;
            if (data.compareTo(parent.data) > 0) {
                parent.rightChild = node;
            } else {
                parent.leftChild = node;
            }
            return node;
        }
    }

    /**
     * 查詢樹中對應(yīng)的結(jié)點(diǎn)
     * <p>
     * data與根結(jié)點(diǎn)比較大小,比根結(jié)點(diǎn)大說明在根節(jié)點(diǎn)的右邊,繼續(xù)data與根節(jié)點(diǎn)的rightChild比大小,以此類推,直到找到與data相等的
     * 比根節(jié)點(diǎn)小在根節(jié)點(diǎn)的左邊,繼續(xù)data與根節(jié)點(diǎn)的leftChild比大小,以此類推,直到找到與data相等的
     * 與根結(jié)點(diǎn)相等則返回根結(jié)點(diǎn)
     *
     * @param data 輸入的值
     * @return
     */
    public TreeNode shearchNode(T data) {

        TreeNode node = root;
        while (node != null) {
            if (data.compareTo(node.data) == 0) {
                return node;
            } else if (data.compareTo(node.data) > 0) {
                node = node.rightChild;
            } else {
                node = node.leftChild;
            }
        }

        return null;
    }

    /**
     * 刪除結(jié)點(diǎn)
     *
     * @param node 要?jiǎng)h除的結(jié)點(diǎn)
     * @return
     */
    public TreeNode delete(TreeNode node) {
        if (node == null) {
            throw new NoSuchElementException();
        } else {
            TreeNode parent = node.parent;
            //1.刪除葉子結(jié)點(diǎn)
            if (node.leftChild == null && node.rightChild == null) {
                //只有一個(gè)結(jié)點(diǎn)
                if (parent == null) {
                    root = null;
                } else {
                    if (parent.leftChild == node) {
                        parent.leftChild = null;
                    } else {
                        parent.rightChild = null;
                    }
                    node.parent = null;
                }
            } else if (node.leftChild != null && node.rightChild == null) {
                //只有左孩子
                /**
                 * 刪除根結(jié)點(diǎn)
                 */
                if (parent == null) {
                    node.leftChild.parent = null;
                    node.leftChild = root;
                } else {
                    //node在parent左邊
                    if (parent.leftChild == node) {
                        node.leftChild.parent = parent;
                        parent.leftChild = node.leftChild;
                    } else {
                        //node在parent右邊
                        node.leftChild.parent = parent;
                        parent.rightChild = node.leftChild;
                    }
                    node.parent = null;
                }
            } else if (node.rightChild != null && node.leftChild == null) {
                //只有右孩子
                /**
                 * 刪除根結(jié)點(diǎn)
                 */
                if (parent == null) {
                    node.rightChild.parent = null;
                    root = node.rightChild;
                } else {
                    //node在parent右邊
                    if (parent.rightChild == node) {
                        node.rightChild.parent = parent;
                        parent.rightChild = node.rightChild;
                    } else {
                        //node在parent左邊
                        node.rightChild.parent = parent;
                        parent.leftChild = node.rightChild;
                    }
                    node.parent = null;
                }
            } else {
                //有左右兩個(gè)孩子
                /**
                 * node的右子樹的左子樹為空,直接補(bǔ)上右子樹
                 */
                if (node.rightChild.leftChild == null) {
                    node.rightChild.leftChild = node.leftChild;
                    if (parent == null) {
                        root = node.leftChild.rightChild;
                    } else {
                        if (parent.leftChild == node) {
                            parent.leftChild = node.leftChild.rightChild;
                        } else {
                            parent.rightChild = node.leftChild.rightChild;
                        }
                    }
                } else {
                    /**
                     * 否則就要補(bǔ)上右子樹的左子樹上最小的一個(gè)
                     */
                    TreeNode leftNode = getMinLeftChild(node.rightChild);
                    TreeNode leftP = leftNode.parent;
                    //1.leftNode的左子樹賦值為node.leftChild
                    leftNode.leftChild = node.leftChild;
                    //2.leftP.leftChild賦值為leftNode.rightChild
                    leftP.leftChild = leftNode.rightChild;
                    //3.leftNode.rightChild = node.rightChild
                    leftNode.rightChild = node.rightChild;
                    //4.leftNode補(bǔ)上去
                    if (parent == null) {
                        root = leftNode;
                    } else {
                        if (parent.leftChild == node) {
                            parent.leftChild = leftNode;
                        } else {
                            parent.rightChild = leftNode;
                        }
                        node.parent = null;
                    }
                }
            }

        }
        return null;
    }

    /**
     * 獲取最小左孩子
     *
     * @param root 根結(jié)點(diǎn)
     * @return
     */
    private TreeNode getMinLeftChild(TreeNode root) {
        if (root == null) {
            return null;
        }
        while (root.leftChild != null) {
            root = root.leftChild;
        }
        return root;
    }

    /**
     * 節(jié)點(diǎn)類
     */
    public static class TreeNode<T extends Comparable> {
        T data;
        TreeNode parent;
        TreeNode leftChild;
        TreeNode rightChild;

        public TreeNode(T data) {
            this.data = data;
            this.parent = null;
            this.leftChild = null;
            this.rightChild = null;
        }

        @Override
        public String toString() {
            return data.toString();
        }
    }

    /**
     * 中序遍歷
     *
     * @param root
     */
    public void midOrderTraverse(TreeNode root) {
        if (root == null) {
            return;
        } else {
            //LDR
            midOrderTraverse(root.leftChild);
            System.out.print(" " + root.toString());
            midOrderTraverse(root.rightChild);
        }
    }

}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末岛请,一起剝皮案震驚了整個(gè)濱河市壳坪,隨后出現(xiàn)的幾起案子独撇,更是在濱河造成了極大的恐慌,老刑警劉巖室抽,帶你破解...
    沈念sama閱讀 212,332評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異葬凳,居然都是意外死亡衅澈,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,508評論 3 385
  • 文/潘曉璐 我一進(jìn)店門窥突,熙熙樓的掌柜王于貴愁眉苦臉地迎上來努溃,“玉大人,你說我怎么就攤上這事阻问∥嗨埃” “怎么了?”我有些...
    開封第一講書人閱讀 157,812評論 0 348
  • 文/不壞的土叔 我叫張陵称近,是天一觀的道長第队。 經(jīng)常有香客問我,道長刨秆,這世上最難降的妖魔是什么凳谦? 我笑而不...
    開封第一講書人閱讀 56,607評論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮衡未,結(jié)果婚禮上晾蜘,老公的妹妹穿的比我還像新娘。我一直安慰自己眠屎,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,728評論 6 386
  • 文/花漫 我一把揭開白布肆饶。 她就那樣靜靜地躺著改衩,像睡著了一般。 火紅的嫁衣襯著肌膚如雪驯镊。 梳的紋絲不亂的頭發(fā)上葫督,一...
    開封第一講書人閱讀 49,919評論 1 290
  • 那天,我揣著相機(jī)與錄音板惑,去河邊找鬼橄镜。 笑死,一個(gè)胖子當(dāng)著我的面吹牛冯乘,可吹牛的內(nèi)容都是我干的洽胶。 我是一名探鬼主播,決...
    沈念sama閱讀 39,071評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼裆馒,長吁一口氣:“原來是場噩夢啊……” “哼姊氓!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起喷好,我...
    開封第一講書人閱讀 37,802評論 0 268
  • 序言:老撾萬榮一對情侶失蹤翔横,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后梗搅,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體禾唁,經(jīng)...
    沈念sama閱讀 44,256評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡效览,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,576評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了荡短。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片丐枉。...
    茶點(diǎn)故事閱讀 38,712評論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖肢预,靈堂內(nèi)的尸體忽然破棺而出矛洞,到底是詐尸還是另有隱情,我是刑警寧澤烫映,帶...
    沈念sama閱讀 34,389評論 4 332
  • 正文 年R本政府宣布沼本,位于F島的核電站,受9級特大地震影響锭沟,放射性物質(zhì)發(fā)生泄漏抽兆。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,032評論 3 316
  • 文/蒙蒙 一族淮、第九天 我趴在偏房一處隱蔽的房頂上張望辫红。 院中可真熱鬧,春花似錦祝辣、人聲如沸贴妻。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,798評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽名惩。三九已至,卻和暖如春孕荠,著一層夾襖步出監(jiān)牢的瞬間娩鹉,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,026評論 1 266
  • 我被黑心中介騙來泰國打工稚伍, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留弯予,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,473評論 2 360
  • 正文 我出身青樓个曙,卻偏偏與公主長得像锈嫩,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子垦搬,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,606評論 2 350

推薦閱讀更多精彩內(nèi)容