94. Binary Tree Inorder Traversal

中序遍歷定義:

先訪問 left subtree,再訪問curNode,再訪問right subtree

思路:

根據(jù)中序遍歷的定義,利用stack來實(shí)現(xiàn)

1. 找到先要訪問的left most node
2. 訪問
3. 如果有right subtree,利用同樣的方法夏醉,再進(jìn)行訪問
4. 如果沒有right subtree,說明當(dāng)前curNode已經(jīng)被fully visited
5. 重復(fù)直到所有節(jié)點(diǎn)被訪問

關(guān)鍵:

  1. stack的意義是涌韩?
    既然stack的方法可以替代recursion的方法畔柔,那么stack一定實(shí)現(xiàn)了recursion中的某個(gè)功能。什么功能呢臣樱?backtrack靶擦!

試想一下,當(dāng)訪問完左子樹之后需要父節(jié)點(diǎn)擎淤,那么怎么才能夠從左子樹的最后一個(gè)被訪問節(jié)點(diǎn)跳到父節(jié)點(diǎn)奢啥?

利用Stack秸仙,可以先將父節(jié)點(diǎn)壓到棧底嘴拢,當(dāng)左子樹的節(jié)點(diǎn)被完全訪問之后,說明棧中已無(wú)左子樹的節(jié)點(diǎn)寂纪,棧頂節(jié)點(diǎn)即為父節(jié)點(diǎn)席吴。

因此,stack中節(jié)點(diǎn)的的意義是:

  • 棧頂的節(jié)點(diǎn)是下一個(gè)要訪問的節(jié)點(diǎn)
  • 位于頂端的比位于底端的節(jié)點(diǎn)得到訪問
  • 已訪問的節(jié)點(diǎn)已出棧
  • 后訪問的節(jié)點(diǎn)可能在棧中或者還未被加到棧中
  1. 如何“重復(fù)直到所有節(jié)點(diǎn)被訪問”捞蛋?
    既然知道了stack的意義孝冒,那么當(dāng)stack為空,則表示所有節(jié)點(diǎn)被訪問拟杉。
while (!stack.empty()) {
  ...
}
  1. 如何“找到left most node”并同時(shí)將父節(jié)點(diǎn)壓到棧中庄涡?
while (cur.left != null) {
       stack.push(cur.left);
       cur = cur.left;
}
//add leftmost
cur = stack.pop();
rst.add(cur.val);
  1. 如果有right subtree?
      // right subtree if possible
      if (cur.right != null) { 
        stack.push(cur.right);
      } 
  1. 如果沒有right subtree搬设?

說明棧頂node的left subtree已經(jīng)被fully visited穴店,此時(shí)棧頂節(jié)點(diǎn)則為下一個(gè)要訪問的點(diǎn)撕捍,此時(shí)可以直接回到while的起始重復(fù)執(zhí)行。

但是有一個(gè)問題泣洞,當(dāng)回到棧頂?shù)墓?jié)點(diǎn)時(shí)忧风,又要經(jīng)過3的循環(huán),即有可能重復(fù)訪問已經(jīng)訪問過的left subtree

解決方法是:

  1. 剪枝cut branch
    因?yàn)楫?dāng)發(fā)現(xiàn)沒有right subtree時(shí)球凰,說明棧頂node的left subtree已經(jīng)被fully visited狮腿,因此可以將其left設(shè)置為null。
// right subtree if possible
      if (cur.right != null) { 
        stack.push(cur.right);
      } else if (!stack.empty()) { //cur subtree is fully visited
        stack.peek().left = null; //cut cur subtree
      }

但改方法更改了輸入呕诉,不好缘厢。

  1. 既然查是否visited嘛,利用個(gè)set唄
while (cur.left != null && !visited.contains(cur.left)) {
     stack.push(cur.left);
     cur = cur.left;
 }

//add leftmost
cur = stack.pop();
rst.add(cur.val);
visited.add(cur);
  1. 巧妙利用在最后使curNode = curNode.right + 找leftmost之前判斷一下curNode是否為null甩挫,將判斷當(dāng)前subtree是否被fully visited的任務(wù)放到找leftmost之前昧绣。
    while (!stack.empty()) {

      if (cur != null) { //when curNode == null, it means cur subtree is fully visited
        // push into stack until the leftmost
        while (cur.left != null) {
          stack.push(cur.left);
          cur = cur.left;
        }
      }

      //add leftmost
      cur = stack.pop();
      rst.add(cur.val);

      // right subtree if possible
      if (cur.right != null) {
        stack.push(cur.right);
      }
      cur = cur.right;

    }

    return rst;
  }
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市捶闸,隨后出現(xiàn)的幾起案子夜畴,更是在濱河造成了極大的恐慌,老刑警劉巖删壮,帶你破解...
    沈念sama閱讀 218,386評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件贪绘,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡央碟,警方通過查閱死者的電腦和手機(jī)税灌,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,142評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來亿虽,“玉大人菱涤,你說我怎么就攤上這事÷迕悖” “怎么了粘秆?”我有些...
    開封第一講書人閱讀 164,704評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)收毫。 經(jīng)常有香客問我攻走,道長(zhǎng),這世上最難降的妖魔是什么此再? 我笑而不...
    開封第一講書人閱讀 58,702評(píng)論 1 294
  • 正文 為了忘掉前任昔搂,我火速辦了婚禮,結(jié)果婚禮上输拇,老公的妹妹穿的比我還像新娘摘符。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,716評(píng)論 6 392
  • 文/花漫 我一把揭開白布逛裤。 她就那樣靜靜地躺著蠢古,像睡著了一般。 火紅的嫁衣襯著肌膚如雪别凹。 梳的紋絲不亂的頭發(fā)上草讶,一...
    開封第一講書人閱讀 51,573評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音炉菲,去河邊找鬼堕战。 笑死,一個(gè)胖子當(dāng)著我的面吹牛拍霜,可吹牛的內(nèi)容都是我干的嘱丢。 我是一名探鬼主播,決...
    沈念sama閱讀 40,314評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼祠饺,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼越驻!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起道偷,我...
    開封第一講書人閱讀 39,230評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤缀旁,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后勺鸦,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體并巍,經(jīng)...
    沈念sama閱讀 45,680評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,873評(píng)論 3 336
  • 正文 我和宋清朗相戀三年换途,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了懊渡。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,991評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡军拟,死狀恐怖剃执,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情懈息,我是刑警寧澤肾档,帶...
    沈念sama閱讀 35,706評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站漓拾,受9級(jí)特大地震影響阁最,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜骇两,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,329評(píng)論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望姜盈。 院中可真熱鬧低千,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,910評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至难审,卻和暖如春瘫拣,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背告喊。 一陣腳步聲響...
    開封第一講書人閱讀 33,038評(píng)論 1 270
  • 我被黑心中介騙來泰國(guó)打工麸拄, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人黔姜。 一個(gè)月前我還...
    沈念sama閱讀 48,158評(píng)論 3 370
  • 正文 我出身青樓拢切,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親秆吵。 傳聞我的和親對(duì)象是個(gè)殘疾皇子淮椰,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,941評(píng)論 2 355

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