二叉查找樹(shù)

一、定義

二叉查找樹(shù)(Binary Search Tree)茉继,又被成為二叉搜索樹(shù)咧叭,是一種特殊的二叉樹(shù)。

二烁竭、特點(diǎn)

1菲茬、若任意節(jié)點(diǎn)的左子樹(shù)不空,則左子樹(shù)上所有結(jié)點(diǎn)的值均小于它的根結(jié)點(diǎn)的值派撕;
2婉弹、若任意節(jié)點(diǎn)的右子樹(shù)不空,則右子樹(shù)上所有結(jié)點(diǎn)的值均大于它的根結(jié)點(diǎn)的值终吼;
3镀赌、任意節(jié)點(diǎn)的左、右子樹(shù)也分別為二叉查找樹(shù)际跪;
4商佛、沒(méi)有鍵值相等的節(jié)點(diǎn)。

三姆打、二叉樹(shù)舉例

二叉查找樹(shù).png

四良姆、二叉查找樹(shù)的實(shí)現(xiàn)

1、節(jié)點(diǎn)類的定義
二叉樹(shù)每個(gè)節(jié)點(diǎn)都有三個(gè)必要的屬性幔戏,左子樹(shù)節(jié)點(diǎn)玛追,右子樹(shù)節(jié)點(diǎn),父親節(jié)點(diǎn)闲延。這里key定義為節(jié)點(diǎn)的值痊剖,同時(shí)確定節(jié)點(diǎn)在樹(shù)中的位置伯复。

public class TNode<T extends Comparable<T>> {
    T key;
    TNode<T> left;
    TNode<T> right;
    TNode<T> parent;
    public TNode(T key, TNode<T> left, TNode<T> right, TNode<T> parent) {
        this.key=key;
        this.left=left;
        this.right=right;
        this.parent=parent;
    }
    public T getKey() {
        return key;
    }
}

2、二叉樹(shù)類的定義

public class BinarySearchTree<T extends Comparable<T>> {
    private TNode<T> root;
    private int size;
    public class TNode<T extends Comparable<T>> {
        T key;
        TNode<T> left;
        TNode<T> right;
        TNode<T> parent;
        public TNode(T key, TNode<T> left, TNode<T> right, TNode<T> parent) {
            this.key=key;
            this.left=left;
            this.right=right;
            this.parent=parent;
        }
        public T getKey() {
            return key;
        }
    }
    public TNode<T> getRoot() {
        return root;
    }
    public void setRoot(TNode<T> root) {
        this.root=root;
    }
    public int getSize() {
        return size;
    }
    public void setSize(int size) {
        this.size=size;
    }
    ......
}

BinarySearchTree是二叉樹(shù)邢笙,它提供了二叉樹(shù)的根節(jié)點(diǎn)root屬性及節(jié)點(diǎn)數(shù)屬性(其他深度等屬性省略)啸如;root是TNode類型,而TNode是二叉查找樹(shù)的節(jié)點(diǎn)氮惯,它是BinarySearchTree的內(nèi)部類叮雳。
3、構(gòu)建一顆二叉查找樹(shù)實(shí)現(xiàn)

 private void insert(BinarySearchTree<T> sourceTree, TNode<T> inNode) {
    int cmp;
    TNode<T> parentNode=null;
    TNode<T> currentNode=sourceTree.root;
    // 優(yōu)先找到待插入的節(jié)點(diǎn)所對(duì)應(yīng)的父親節(jié)點(diǎn)
    while(null != currentNode) {
        parentNode=currentNode;
        cmp=inNode.key.compareTo(currentNode.key);// 將待插入節(jié)點(diǎn)值和當(dāng)前節(jié)點(diǎn)值進(jìn)行比較
        if(cmp < 0) {// 如果比當(dāng)前節(jié)點(diǎn)值小
            currentNode=currentNode.left;// 將當(dāng)前節(jié)點(diǎn)的左子樹(shù)當(dāng)作下一次要比較的父親節(jié)點(diǎn)
        } else {// 如果比當(dāng)前節(jié)點(diǎn)值小
            currentNode=currentNode.right;
        }
    }
    // 以上找到了父親節(jié)點(diǎn),再將inNode的父親節(jié)點(diǎn)設(shè)置為parentNode
    inNode.parent=parentNode;
    if(null == parentNode) {// 如果parentNode仍然為空,則將inNode設(shè)置為root節(jié)點(diǎn)
        sourceTree.root=inNode;
    } else {
        cmp=inNode.key.compareTo(parentNode.key);
        if(cmp < 0) { //待插入節(jié)點(diǎn)值比父親節(jié)點(diǎn)值小
            parentNode.left=inNode; //將待插入節(jié)點(diǎn)設(shè)置為父親節(jié)點(diǎn)的左子樹(shù)
            size++;
        } else {//待插入節(jié)點(diǎn)值比父親節(jié)點(diǎn)值大
            parentNode.right=inNode;//將待插入節(jié)點(diǎn)設(shè)置為父親節(jié)點(diǎn)的右子樹(shù)
            size++;
        }
    }
}

public void insert(T key) {
    TNode<T> z=new TNode<T>(key, null, null, null);
    // 如果新建結(jié)點(diǎn)失敗妇汗,則返回帘不。
    if(z != null) {
        insert(this, z);
    }
}

插入新的節(jié)點(diǎn)其實(shí)很簡(jiǎn)單,分兩步杨箭;第一步:找到父親節(jié)點(diǎn)寞焙;第二步,比較待插入節(jié)點(diǎn)值和父親節(jié)點(diǎn)值互婿,如果比父親節(jié)點(diǎn)值小捣郊,則被設(shè)置為父親節(jié)點(diǎn)的左子樹(shù),反之設(shè)置為父親節(jié)點(diǎn)的右子樹(shù)慈参。
4呛牲、二叉查找樹(shù)遍歷實(shí)現(xiàn)

//前序遍歷
private void preOrder(TNode<T> node) {
    if(null != node) {
        System.out.print(node.key + " ");
        preOrder(node.left);
        preOrder(node.right);
    }
}
public void preOrder() {
    preOrder(root);
}
//后序遍歷
private void postOrder(TNode<T> node) {
    if(null != node) {
        postOrder(node.left);
        postOrder(node.right);
        System.out.print(node.key + " ");
    }
}
public void postOrder() {
    postOrder(root);
}
//中序遍歷
private void inOrder(TNode<T> node) {
    if(null != node) {
        postOrder(node.left);
        System.out.print(node.key + " ");
        postOrder(node.right);
    }
}
public void inOrder() {
    inOrder(root);
}

單元測(cè)試實(shí)現(xiàn):

public static void main(String[] args) {
    int[] arr={1, 5, 4, 3, 2, 6};
    int i, ilen;
    BinarySearchTree<Integer> tree=new BinarySearchTree<Integer>();
    System.out.print("依次添加: ");
    ilen=arr.length;
    for(i=0; i < ilen; i++) {
        System.out.print(arr[i] + " ");
        tree.insert(arr[i]);
    }
    System.out.print("\n前序遍歷: ");
    tree.preOrder();
    System.out.print("\n中序遍歷: ");
    tree.inOrder();
    System.out.print("\n后序遍歷: ");
    tree.postOrder();
}
}

控制臺(tái)打印結(jié)果:

依次添加: 1 5 4 3 2 6 
前序遍歷: 1 5 4 3 2 6 
中序遍歷: 1 2 3 4 6 5 
后序遍歷: 2 3 4 6 5 1 

構(gòu)建過(guò)程如下圖:

構(gòu)建流程.jpg

5、元素查找

private TNode<T> search(TNode<T> node, T key) {
    if(null == node) {
        return node;
    }
    int cmp=key.compareTo(node.key);
    if(cmp < 0) {
        return search(node.left, key);
    } else if(cmp > 0) {
        return search(node.right, key);
    }
    return node;
}
public TNode<T> search(T key) {
    return search(root, key);
}

5驮配、查找最大值與最小值

//查詢最大值
private TNode<T> max(TNode<T> tree) {
    if(null == tree) {
        return tree;
    }
    while(null != tree.right) {
        tree=tree.right;
    }
    return tree;
}
public T max() {
    TNode<T> node=max(root);
    if(null != node) {
        return node.key;
    }
    return null;
}

//查詢最小值
private TNode<T> min(TNode<T> tree) {
    if(null == tree) {
        return tree;
    }
    while(null != tree.left) {
        tree=tree.left;
    }
    return tree;
}
public T min() {
    TNode<T> node=min(root);
    if(null != node) {
        return node.key;
    }
    return null;
}

--未完待續(xù)--

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末娘扩,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子壮锻,更是在濱河造成了極大的恐慌琐旁,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,084評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件猜绣,死亡現(xiàn)場(chǎng)離奇詭異灰殴,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)途事,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,623評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門(mén)验懊,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)擅羞,“玉大人尸变,你說(shuō)我怎么就攤上這事〖跚危” “怎么了召烂?”我有些...
    開(kāi)封第一講書(shū)人閱讀 163,450評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)娃承。 經(jīng)常有香客問(wèn)我奏夫,道長(zhǎng)怕篷,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,322評(píng)論 1 293
  • 正文 為了忘掉前任酗昼,我火速辦了婚禮廊谓,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘麻削。我一直安慰自己蒸痹,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,370評(píng)論 6 390
  • 文/花漫 我一把揭開(kāi)白布呛哟。 她就那樣靜靜地躺著叠荠,像睡著了一般。 火紅的嫁衣襯著肌膚如雪扫责。 梳的紋絲不亂的頭發(fā)上榛鼎,一...
    開(kāi)封第一講書(shū)人閱讀 51,274評(píng)論 1 300
  • 那天,我揣著相機(jī)與錄音鳖孤,去河邊找鬼者娱。 笑死,一個(gè)胖子當(dāng)著我的面吹牛苏揣,可吹牛的內(nèi)容都是我干的肺然。 我是一名探鬼主播,決...
    沈念sama閱讀 40,126評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼腿准,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼际起!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起吐葱,我...
    開(kāi)封第一講書(shū)人閱讀 38,980評(píng)論 0 275
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤街望,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后弟跑,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體灾前,經(jīng)...
    沈念sama閱讀 45,414評(píng)論 1 313
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,599評(píng)論 3 334
  • 正文 我和宋清朗相戀三年孟辑,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了哎甲。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,773評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡饲嗽,死狀恐怖炭玫,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情貌虾,我是刑警寧澤吞加,帶...
    沈念sama閱讀 35,470評(píng)論 5 344
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響衔憨,放射性物質(zhì)發(fā)生泄漏叶圃。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,080評(píng)論 3 327
  • 文/蒙蒙 一践图、第九天 我趴在偏房一處隱蔽的房頂上張望掺冠。 院中可真熱鬧,春花似錦码党、人聲如沸赫舒。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,713評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)接癌。三九已至,卻和暖如春扣讼,著一層夾襖步出監(jiān)牢的瞬間缺猛,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,852評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工椭符, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留荔燎,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,865評(píng)論 2 370
  • 正文 我出身青樓销钝,卻偏偏與公主長(zhǎng)得像有咨,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子蒸健,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,689評(píng)論 2 354