每日一題之《劍指offer》62,63題

第62題:二叉搜索樹的第k個(gè)節(jié)點(diǎn)

難易度:??

題目描述:
給定一棵二叉搜索樹,請(qǐng)找出其中的第k小的結(jié)點(diǎn)。
例如, (5,3急侥,7砌滞,2侮邀,4,6贝润,8)    中绊茧,按結(jié)點(diǎn)數(shù)值大小順序第三小結(jié)點(diǎn)的值為4。

解析:
二叉搜索樹的特點(diǎn)就是如果中序遍歷二叉搜索樹打掘,打印出來的節(jié)點(diǎn)value值會(huì)按照從小到大的順序進(jìn)行排列华畏,可以利用這一點(diǎn),用一個(gè)變量去標(biāo)記index值尊蚁,當(dāng)index等于k時(shí)亡笑,返回該節(jié)點(diǎn)即可。
代碼如下:

/*
public class TreeNode {
    int val = 0;
    TreeNode left = null;
    TreeNode right = null;

    public TreeNode(int val) {
        this.val = val;

    }

}
*/
import java.util.Stack;
public class Solution {
    TreeNode KthNode(TreeNode pRoot, int k){
        if(pRoot == null || k < 0){
            return null;
        }
        int index = 0;
        Stack<TreeNode> stack = new Stack<>();
        while(pRoot != null || !stack.isEmpty()){
            if(pRoot != null){
                stack.push(pRoot);
                pRoot = pRoot.left;
            }else{
                pRoot = stack.pop();
                index++;
                // do sth
                if(index == k){
                    return pRoot;
                }
                pRoot = pRoot.right;
            }
        }
        return null;
    }
}

第63題:數(shù)據(jù)流中的中位數(shù)

難易度:???

如何得到一個(gè)數(shù)據(jù)流中的中位數(shù)横朋?
如果從數(shù)據(jù)流中讀出奇數(shù)個(gè)數(shù)值仑乌,那么中位數(shù)就是所有數(shù)值排序之后位于中間的數(shù)值。
如果從數(shù)據(jù)流中讀出偶數(shù)個(gè)數(shù)值,那么中位數(shù)就是所有數(shù)值排序之后中間兩個(gè)數(shù)的平均值晰甚。
我們使用Insert()方法讀取數(shù)據(jù)流衙传,使用GetMedian()方法獲取當(dāng)前讀取數(shù)據(jù)的中位數(shù)。

如果你用一個(gè)數(shù)組去接收流吐出的數(shù)字厕九,那么收集流吐出的數(shù)字很簡(jiǎn)單蓖捶,但是如果要獲取中位數(shù)的話就要對(duì)數(shù)組進(jìn)行一次排序,如果要求流每次吐出一個(gè)數(shù)字就獲取一次中位數(shù)扁远,那么就要頻繁地對(duì)數(shù)組進(jìn)行排序操作俊鱼。解決的方法就是使用堆結(jié)構(gòu)來解決這個(gè)問題:設(shè)計(jì)一個(gè)最大堆,和一個(gè)最小堆穿香,每次流吐出數(shù)字的時(shí)候都和最大堆的堆頂作比較亭引,如果小于等于最大堆堆頂就放入最大堆,如果大于最大堆的堆頂則放入最小堆皮获。同時(shí)我們還需要保證最大堆的數(shù)據(jù)量不能比最小堆中的數(shù)據(jù)量大超過1焙蚓,反之亦是如此,也就是要保持最大堆數(shù)據(jù)量N和最小堆的數(shù)據(jù)量M的關(guān)系為 |N-M| <= 1,如何保持這種關(guān)系呢?一旦破壞了這種關(guān)系的平衡洒宝,我們就讓最大堆的頂部或者最小堆的頂部彈出购公,讓這個(gè)最大的數(shù)字進(jìn)入到最小堆里面,或者是讓最小堆的最小的數(shù)字加到最大堆里面雁歌,如圖示意:


image.png

假設(shè)這個(gè)數(shù)據(jù)流吐出的數(shù)字依次是 5->6->4->7->3->8->1
左側(cè)為最大堆數(shù)字?jǐn)?shù)量為N宏浩,右側(cè)為最小堆數(shù)字?jǐn)?shù)量為M,沒有破壞過 |N-M| <= 1的規(guī)定,如果這個(gè)時(shí)候數(shù)據(jù)流吐出的數(shù)字為0靠瞎,那么就要add到最大堆里面比庄,這個(gè)時(shí)候N-M = 2破壞了兩堆數(shù)量上的平衡,所以要把最大堆的堆頂彈入到最小堆中去乏盐,操作如下:

首先將數(shù)字0 add 到最大堆中


image.png

交換最大堆首尾數(shù)字


image.png

將最大堆末尾數(shù)字(5) add 到最小堆
image.png

接下來就是heapify操作了佳窑,對(duì)最大堆中的堆頂0進(jìn)行heapify,首先要判斷它的兩個(gè)孩子誰更大,誰大跟誰交換父能,直到滿足最大堆的結(jié)構(gòu)為止


image.png

本題的思路就是使用一個(gè)最大堆和一個(gè)最小堆解決神凑,優(yōu)先級(jí)隊(duì)列PriorityQueue即可以滿足本題的求解,代碼如下:
import java.util.Comparator;
import java.util.PriorityQueue;

public class Solution {
    private PriorityQueue<Integer> minHeap = new PriorityQueue<>(new MinHeapCompare());
    private PriorityQueue<Integer> maxHeap = new PriorityQueue<>(new MaxHeapCompare());
    public static class MinHeapCompare implements Comparator<Integer> {
        @Override
        public int compare(Integer o1, Integer o2) {
            return o1 - o2;
        }
    }

    public static class MaxHeapCompare implements Comparator<Integer>{
        @Override
        public int compare(Integer o1,Integer o2){
            return o2 - o1;
        }
    }
    
    private void modify(){
        if(maxHeap.size() == minHeap.size() + 2){
            minHeap.add(maxHeap.poll());
        }
        if(minHeap.size() == maxHeap.size() + 2){
            maxHeap.add(minHeap.poll());
        }
    }
    
    public void Insert(Integer num) {
        if(maxHeap.isEmpty()){
            maxHeap.add(num);
            return;
        }
        if(num <= maxHeap.peek()){
            maxHeap.add(num);
        }else{
            if(minHeap.isEmpty()){
                minHeap.add(num);
                return;
            }
            if(num < minHeap.peek()){
                maxHeap.add(num);
            }else{
                minHeap.add(num);
            }
        }
        modify();
    }

    public Double GetMedian() {
        if(maxHeap.isEmpty() && minHeap.isEmpty()){
            return null;
        }
        int maxHeapSize = maxHeap.size();
        int minHeapSize = minHeap.size();
        int maxHeapHead = maxHeapSize == 0 ? 0 : maxHeap.peek();
        int minHeapHead = minHeapSize == 0 ? 0 :minHeap.peek();


        if(maxHeapSize == minHeapSize){
            return ((double)(maxHeapHead + minHeapHead))/2;
        }
        return maxHeapSize > minHeapSize ? (double)maxHeapHead : (double)minHeapHead;
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末何吝,一起剝皮案震驚了整個(gè)濱河市溉委,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌爱榕,老刑警劉巖瓣喊,帶你破解...
    沈念sama閱讀 219,490評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異黔酥,居然都是意外死亡藻三,警方通過查閱死者的電腦和手機(jī)八匠,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,581評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來趴酣,“玉大人梨树,你說我怎么就攤上這事♂” “怎么了抡四?”我有些...
    開封第一講書人閱讀 165,830評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)仗谆。 經(jīng)常有香客問我指巡,道長(zhǎng),這世上最難降的妖魔是什么隶垮? 我笑而不...
    開封第一講書人閱讀 58,957評(píng)論 1 295
  • 正文 為了忘掉前任藻雪,我火速辦了婚禮,結(jié)果婚禮上狸吞,老公的妹妹穿的比我還像新娘勉耀。我一直安慰自己,他們只是感情好蹋偏,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,974評(píng)論 6 393
  • 文/花漫 我一把揭開白布便斥。 她就那樣靜靜地躺著,像睡著了一般威始。 火紅的嫁衣襯著肌膚如雪枢纠。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,754評(píng)論 1 307
  • 那天黎棠,我揣著相機(jī)與錄音晋渺,去河邊找鬼。 笑死脓斩,一個(gè)胖子當(dāng)著我的面吹牛木西,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播俭厚,決...
    沈念sama閱讀 40,464評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼户魏,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼驶臊!你這毒婦竟也來了挪挤?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,357評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤关翎,失蹤者是張志新(化名)和其女友劉穎扛门,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體纵寝,經(jīng)...
    沈念sama閱讀 45,847評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡论寨,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,995評(píng)論 3 338
  • 正文 我和宋清朗相戀三年星立,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片葬凳。...
    茶點(diǎn)故事閱讀 40,137評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡绰垂,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出火焰,到底是詐尸還是另有隱情劲装,我是刑警寧澤,帶...
    沈念sama閱讀 35,819評(píng)論 5 346
  • 正文 年R本政府宣布昌简,位于F島的核電站占业,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏纯赎。R本人自食惡果不足惜谦疾,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,482評(píng)論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望犬金。 院中可真熱鬧念恍,春花似錦、人聲如沸晚顷。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,023評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽音同。三九已至词爬,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間权均,已是汗流浹背顿膨。 一陣腳步聲響...
    開封第一講書人閱讀 33,149評(píng)論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留叽赊,地道東北人恋沃。 一個(gè)月前我還...
    沈念sama閱讀 48,409評(píng)論 3 373
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像必指,于是被迫代替她去往敵國和親囊咏。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,086評(píng)論 2 355

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

  • 下面代碼的輸出是什么 因?yàn)閟izeof(data1)是求數(shù)組的大小,每個(gè)整數(shù)占4個(gè)字節(jié)葛家;第二個(gè)是因?yàn)橹羔樥?個(gè)字節(jié)...
    世界上的一道風(fēng)閱讀 344評(píng)論 0 0
  • 第1章 面試的流程 編程時(shí)應(yīng)注意的三點(diǎn): 思考清楚再開始編碼户辞; 良好的代碼命名和縮進(jìn)對(duì)齊; 能夠單元測(cè)試癞谒; 現(xiàn)場(chǎng)面...
    codingXue閱讀 488評(píng)論 5 0
  • 快速排序 荷蘭國旗問題(Dutch National Flag Problem) 思路:給定一個(gè)無序數(shù)組[4,5,...
    憨憨二師兄閱讀 607評(píng)論 0 1
  • 貪心算法是指,在對(duì)問題求解時(shí)双仍,將整體的問題劃分為一系列局部問題枢希,對(duì)局部的問題求出最優(yōu)解,再通過數(shù)學(xué)歸納法證明每一步...
    憨憨二師兄閱讀 395評(píng)論 0 0
  • 劍指offer61到66題總覽: 序列化二叉樹 二叉搜索樹的第k個(gè)結(jié)點(diǎn) 數(shù)據(jù)流中的中位數(shù) 滑動(dòng)窗口的最大值 矩陣中...
    藍(lán)白絳閱讀 255評(píng)論 0 0