算法題心得 - 樹

上一篇文章介紹了鏈表猴誊,這篇文章繼續(xù)分享算法心得阁将,即另一個(gè)重要的數(shù)據(jù)結(jié)構(gòu)——樹膏秫。

樹的特點(diǎn)

樹和鏈表相對(duì)比,會(huì)更加復(fù)雜一些做盅。實(shí)際上鏈表中節(jié)點(diǎn)有一個(gè)指針指向下一個(gè)節(jié)點(diǎn)缤削,而在樹中窘哈,有兩個(gè)指針,分別指向左亭敢,右兩個(gè)孩子節(jié)點(diǎn)滚婉。這種對(duì)應(yīng)關(guān)系,就不是鏈表中的一對(duì)一了帅刀,在樹中让腹,對(duì)應(yīng)關(guān)系就是一對(duì)多了。如果說鏈表可以理解為一維的世界扣溺,則樹就是二維的世界了骇窍。特別的,樹這個(gè)二維世界有一個(gè)非常重要的節(jié)點(diǎn)就是根節(jié)點(diǎn)锥余,這個(gè)節(jié)點(diǎn)可以認(rèn)為是樹的特征節(jié)點(diǎn)腹纳。因此解決樹的問題,找準(zhǔn)根節(jié)點(diǎn)是非常重要哈恰。

在23種設(shè)計(jì)模式中只估,組合模式就是運(yùn)用了樹這種數(shù)據(jù)結(jié)構(gòu)志群。組合模式的意思就是操作樹根這一個(gè)元素着绷,好像同時(shí)操作整個(gè)樹一樣。在實(shí)際的應(yīng)用中锌云,樹可以理解為一種層級(jí)結(jié)構(gòu)荠医,比如界面,windows可能是樹根桑涎,樹根會(huì)有視圖彬向,視圖又可分為用戶自定義的各種視圖控件,整個(gè)的ui界面攻冷,可以理解為一棵完整的樹娃胆,而操作樹根,比如windows接收到了點(diǎn)擊事件等曼,就可以將這個(gè)事件分發(fā)給這課樹的所有節(jié)點(diǎn)里烦。這就是組合模式。

樹的一個(gè)重要的特點(diǎn)禁谦,就是有一定的遞歸性胁黑,因此,在解決樹相關(guān)的問題時(shí)州泊,遞歸的方法丧蘸,顯然是首選。另外遥皂,結(jié)合樹中最重要的要點(diǎn)力喷,就是找到樹的特征也就是樹根刽漂。把握好這兩個(gè)方面,很多樹相關(guān)的問題就可以解決了弟孟。

樹的遍歷

樹既然具有二維性爽冕,一個(gè)重要的課題就是將二維轉(zhuǎn)換為一維,也就是樹的遍歷披蕉。樹的遍歷總共分為三種颈畸,前序遍歷,中序遍歷和后序遍歷没讲。這里的前眯娱,中和后都是說的根,也就是根遍歷的順序爬凑。實(shí)際上在樹的遍歷中徙缴,永遠(yuǎn)是先左節(jié)點(diǎn),后右節(jié)點(diǎn)的嘁信。但是關(guān)鍵是根節(jié)點(diǎn)于样。如果先是根節(jié)點(diǎn),然后左節(jié)點(diǎn)潘靖,最后右節(jié)點(diǎn)穿剖,就是前序遍歷,也就是根在前面遍歷卦溢。如果先是左節(jié)點(diǎn)糊余,然后根節(jié)點(diǎn),最后右節(jié)點(diǎn)单寂,就是中序遍歷贬芥,也就是根在中間遍歷。如果先是左節(jié)點(diǎn)宣决,然后右節(jié)點(diǎn)蘸劈,最后根節(jié)點(diǎn),就是后序遍歷尊沸,也就是根在最后遍歷威沫。

樹的三種遍歷方式,可以將二維的樹椒丧,轉(zhuǎn)換為一維的數(shù)組壹甥。前序,中序和后序是樹的一維形態(tài)壶熏。二維可以轉(zhuǎn)換為一維句柠,那么一維是否可以轉(zhuǎn)換為二維呢?這實(shí)際上就是一個(gè)面試題,我們將在下面詳細(xì)分析溯职。

樹的典型算法題

面試題7:重建二叉樹

題目:輸入某二叉樹的前序遍歷和中序遍歷的結(jié)果精盅,請(qǐng)重建該二叉樹。假設(shè)輸入的前序遍歷和中序遍歷的結(jié)果中都不含重復(fù)的數(shù)字谜酒。

這是一道典型的由一維的遍歷重構(gòu)出二維的樹叹俏。由于不含重復(fù)的數(shù)字,因此這種重構(gòu)是可能的僻族≌吵郏可以這樣理解,用兩個(gè)一維數(shù)據(jù)述么,重構(gòu)出二維數(shù)據(jù)蝌数,這是可行的。題目要求用前序遍歷和中序遍歷重建二叉樹度秘,而樹的結(jié)構(gòu)顶伞,最重要的兩個(gè)元素就是根節(jié)點(diǎn)以及遞歸。一提到遞歸剑梳,我們就需要處理異常唆貌,終止以及問題化簡(jiǎn)。有了這些基本的武器后垢乙,我們看看如何解決這個(gè)問題锨咙。

解決樹的問題最核心的是找到根節(jié)點(diǎn),如何找到根節(jié)點(diǎn)呢侨赡?很簡(jiǎn)單蓖租,前序遍歷中就有根節(jié)點(diǎn)了粱侣。前序遍歷之所以稱之為前序羊壹,就是因?yàn)槭鞘紫缺闅v根節(jié)點(diǎn)。因此齐婴,前序遍歷可以理解為 根節(jié)點(diǎn) + 左子樹節(jié)點(diǎn) + 右子樹節(jié)點(diǎn)油猫。如果我們想利用遞歸的思想,就必須對(duì)題目進(jìn)行簡(jiǎn)化柠偶,簡(jiǎn)化的方法就是將樹的問題轉(zhuǎn)換為左子樹和右子樹的問題情妖,這樣,就找到了問題的子問題诱担。這里我們根據(jù)前序遍歷的特點(diǎn)毡证,可以得到根節(jié)點(diǎn),但是左子樹和右子樹的邊界我們是不清楚的蔫仙。因此料睛,想到利用另一個(gè)已知,即中序遍歷。中序遍歷的遍歷方式是 左子樹節(jié)點(diǎn) + 根節(jié)點(diǎn) + 右子樹節(jié)點(diǎn)恤煞。從前序遍歷中屎勘,我們已經(jīng)找到了根節(jié)點(diǎn),因此從中序遍歷中就可以確定了根節(jié)點(diǎn)的位置居扒,根節(jié)點(diǎn)的左邊就是左子樹的中序遍歷概漱,根節(jié)點(diǎn)的右邊就是右子樹的中序遍歷。特別的喜喂,我們可以得到左子樹和右子樹的元素的個(gè)數(shù)瓤摧。確定了左子樹和右子樹元素的個(gè)數(shù)后,我們又可以通過前序遍歷玉吁,確定左子樹和右子樹的分割邊界姻灶。這樣,我們就可以得到左右子樹的前序遍歷诈茧。

分析到這里产喉,這個(gè)問題基本上就已經(jīng)解開了,我們成功的將問題轉(zhuǎn)換為子問題敢会,即我們可以分別得到根節(jié)點(diǎn)曾沈,以及左右子樹的前序和中序遍歷。通過遞歸鸥昏,我們就可以將這棵樹重建起來塞俱。

面試題32:從上到下打印二叉樹

題目:不分行從上到下打印二叉樹
從上到下打印出二叉樹的每個(gè)節(jié)點(diǎn),同一層的節(jié)點(diǎn)按照從左到右的順序打印吏垮。

為什么想分享這樣一道面試題呢障涯?因?yàn)檫@道面試題,體現(xiàn)了樹這個(gè)數(shù)據(jù)結(jié)構(gòu)可以和其他一些數(shù)據(jù)結(jié)構(gòu)相組合膳汪。我們常用的數(shù)據(jù)結(jié)構(gòu)還有什么呢唯蝶?實(shí)際上還有先進(jìn)先出的隊(duì)列,以及先進(jìn)后出的棧遗嗽。有時(shí)粘我,解決樹相關(guān)的問題,也需要輔助隊(duì)列痹换,或者輔助棧征字,比如這道算法題就是一個(gè)典型。

這道題考察了一種特殊的樹的遍歷方式娇豫,這種方式既不是前序遍歷匙姜,也不是中序遍歷,更不是后序遍歷冯痢。而是一種自定義的遍歷方式氮昧,是按層級(jí)遍歷或详,也就是自上而下的遍歷。這種遍歷方式似乎破壞了樹的遞歸的特性郭计,那么我們有沒有什么好方法呢霸琴?

實(shí)際上這種遍歷方式,有些道家的一生二昭伸,二生三梧乘,三生萬物的意思了÷睿或者更形象些选调,就是太極生兩儀,兩儀生四象灵份,四象生八卦仁堪。這個(gè)太極,就是樹的根結(jié)構(gòu)填渠。既然有“生”弦聂,就要有容器去容納那些生出來的節(jié)點(diǎn),由于這種“生”的方式氛什,符合先進(jìn)先出的特點(diǎn)莺葫,因此我們需要一個(gè)輔助隊(duì)列。首先我們把樹根扔進(jìn)這個(gè)隊(duì)列中枪眉,然后開始處理隊(duì)列捺檬。處理隊(duì)列的時(shí)候,我們首先將隊(duì)頭的那個(gè)節(jié)點(diǎn)的左右子節(jié)點(diǎn)扔進(jìn)隊(duì)列中贸铜,然后將這個(gè)節(jié)點(diǎn)彈出隊(duì)列堡纬,打印其信息。如此操作蒿秦,就是符合題目的遍歷方式了烤镐。

通過這個(gè)面試題,我們可以很清晰的看到渤早,樹這種數(shù)據(jù)結(jié)構(gòu)有時(shí)可以和其他基本數(shù)據(jù)結(jié)構(gòu)相配合职车,比如這里,使用了一個(gè)輔助的隊(duì)列鹊杖。

小結(jié)

樹的算法問題解決思路大多都是遞歸思想,因?yàn)檫@是由他的結(jié)構(gòu)所決定的扛芽。特別的骂蓖,對(duì)于樹而言,最關(guān)鍵的就是找到他的根節(jié)點(diǎn)川尖,找到了根節(jié)點(diǎn)登下,同時(shí)就可以確定其左子樹和右子樹,因此樹的問題可以轉(zhuǎn)換為左子樹和右子樹的子問題,因此可以采用遞歸的思想被芳。另外要注意的是缰贝,樹有可能會(huì)和其他基本數(shù)據(jù)結(jié)構(gòu)相配合,比如棧和隊(duì)列畔濒。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末剩晴,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子侵状,更是在濱河造成了極大的恐慌赞弥,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,591評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件趣兄,死亡現(xiàn)場(chǎng)離奇詭異绽左,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)艇潭,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,448評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門拼窥,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人蹋凝,你說我怎么就攤上這事闯团。” “怎么了仙粱?”我有些...
    開封第一講書人閱讀 162,823評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵房交,是天一觀的道長。 經(jīng)常有香客問我伐割,道長候味,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,204評(píng)論 1 292
  • 正文 為了忘掉前任隔心,我火速辦了婚禮白群,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘硬霍。我一直安慰自己帜慢,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,228評(píng)論 6 388
  • 文/花漫 我一把揭開白布唯卖。 她就那樣靜靜地躺著粱玲,像睡著了一般。 火紅的嫁衣襯著肌膚如雪拜轨。 梳的紋絲不亂的頭發(fā)上抽减,一...
    開封第一講書人閱讀 51,190評(píng)論 1 299
  • 那天,我揣著相機(jī)與錄音橄碾,去河邊找鬼卵沉。 笑死颠锉,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的史汗。 我是一名探鬼主播琼掠,決...
    沈念sama閱讀 40,078評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼停撞!你這毒婦竟也來了瓷蛙?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,923評(píng)論 0 274
  • 序言:老撾萬榮一對(duì)情侶失蹤怜森,失蹤者是張志新(化名)和其女友劉穎速挑,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體副硅,經(jīng)...
    沈念sama閱讀 45,334評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡姥宝,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,550評(píng)論 2 333
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了恐疲。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片腊满。...
    茶點(diǎn)故事閱讀 39,727評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖培己,靈堂內(nèi)的尸體忽然破棺而出碳蛋,到底是詐尸還是另有隱情,我是刑警寧澤省咨,帶...
    沈念sama閱讀 35,428評(píng)論 5 343
  • 正文 年R本政府宣布肃弟,位于F島的核電站,受9級(jí)特大地震影響零蓉,放射性物質(zhì)發(fā)生泄漏笤受。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,022評(píng)論 3 326
  • 文/蒙蒙 一敌蜂、第九天 我趴在偏房一處隱蔽的房頂上張望箩兽。 院中可真熱鬧,春花似錦章喉、人聲如沸汗贫。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,672評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽落包。三九已至,卻和暖如春撞反,著一層夾襖步出監(jiān)牢的瞬間妥色,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,826評(píng)論 1 269
  • 我被黑心中介騙來泰國打工遏片, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留嘹害,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,734評(píng)論 2 368
  • 正文 我出身青樓吮便,卻偏偏與公主長得像笔呀,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子髓需,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,619評(píng)論 2 354

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