JS樹結(jié)構(gòu)操作


一姜胖、遍歷樹結(jié)構(gòu)

1. 樹結(jié)構(gòu)介紹

JS中樹結(jié)構(gòu)一般是類似于這樣的結(jié)構(gòu):

let tree = [
  {
    id: '1',
    title: '節(jié)點1',
    children: [
      {
        id: '1-1',
        title: '節(jié)點1-1'
      },
      {
        id: '1-2',
        title: '節(jié)點1-2'
      }
    ]
  },
  {
    id: '2',
    title: '節(jié)點2',
    children: [
      {
        id: '2-1',
        title: '節(jié)點2-1'
      }
    ]
  }
]

為了更通用,可以用存儲了樹根節(jié)點的列表表示一個樹形結(jié)構(gòu)怠蹂,每個節(jié)點的children屬性(如果有)是一顆子樹,如果沒有children屬性或者children長度為0,則表示該節(jié)點為葉子節(jié)點氢烘。

2. 樹結(jié)構(gòu)遍歷方法介紹

樹結(jié)構(gòu)的常用場景之一就是遍歷,而遍歷又分為廣度優(yōu)先遍歷家厌、深度優(yōu)先遍歷播玖。其中深度優(yōu)先遍歷是可遞歸的,而廣度優(yōu)先遍歷是非遞歸的饭于,通常用循環(huán)來實現(xiàn)蜀踏。深度優(yōu)先遍歷又分為先序遍歷、后序遍歷掰吕,二叉樹還有中序遍歷果覆,實現(xiàn)方法可以是遞歸,也可以是循環(huán)殖熟。


JS遍歷樹結(jié)構(gòu)

廣度優(yōu)先和深度優(yōu)先的概念很簡單局待,區(qū)別如下:

  • 深度優(yōu)先,訪問完一顆子樹再去訪問后面的子樹菱属,而訪問子樹的時候钳榨,先訪問根再訪問根的子樹,稱為先序遍歷纽门;先訪問子樹再訪問根薛耻,稱為后序遍歷。
  • 廣度優(yōu)先膜毁,即訪問樹結(jié)構(gòu)的第n+1層前必須先訪問完第n層
3. 廣度優(yōu)先遍歷的實現(xiàn)

廣度優(yōu)先的思路是昭卓,維護(hù)一個隊列,隊列的初始值為樹結(jié)構(gòu)根節(jié)點組成的列表瘟滨,重復(fù)執(zhí)行以下步驟直到隊列為空:

  • 取出隊列中的第一個元素候醒,進(jìn)行訪問相關(guān)操作,然后將其后代元素(如果有)全部追加到隊列最后杂瘸。
    下面是代碼實現(xiàn)倒淫,類似于數(shù)組的forEach遍歷,我們將數(shù)組的訪問操作交給調(diào)用者自定義败玉,即一個回調(diào)函數(shù):
// 廣度優(yōu)先
function treeForeach (tree, func) {
  let node, list = [...tree]
  while (node = list.shift()) {
    func(node)
    node.children && list.push(...node.children)
  }
}

很簡單吧敌土,用上述數(shù)據(jù)測試一下看看:

treeForeach(tree, node => { console.log(node.title) })

輸出镜硕,可以看到第一層所有元素都在第二層元素前輸出:

> 節(jié)點1
> 節(jié)點2
> 節(jié)點1-1
> 節(jié)點1-2
> 節(jié)點2-1
4. 深度優(yōu)先遍歷的遞歸實現(xiàn)

先序遍歷,三五行代碼返干,太簡單兴枯,不過多描述了:

function treeForeach (tree, func) {
  tree.forEach(data => {
    func(data)
    data.children && treeForeach(data.children, func) // 遍歷子樹
  })
}

后序遍歷,與先序遍歷思想一致矩欠,代碼也及其相似财剖,只不過調(diào)換一下節(jié)點遍歷和子樹遍歷的順序:

function treeForeach (tree, func) {
  tree.forEach(data => {
    data.children && treeForeach(data.children, func) // 遍歷子樹
    func(data)
  })
}

測試:

treeForeach(tree, node => { console.log(node.title) })

輸出:

// 先序遍歷
> 節(jié)點1
> 節(jié)點1-1
> 節(jié)點1-2
> 節(jié)點2
> 節(jié)點2-1
// 后序遍歷
> 節(jié)點1-1
> 節(jié)點1-2
> 節(jié)點1
> 節(jié)點2-1
> 節(jié)點2

5. 深度優(yōu)先循環(huán)實現(xiàn)

先序遍歷與廣度優(yōu)先循環(huán)實現(xiàn)類似,要維護(hù)一個隊列癌淮,不同的是子節(jié)點不追加到隊列最后躺坟,而是加到隊列最前面:

function treeForeach (tree, func) {
  let node, list = [...tree]
  while (node = list.shift()) {
    func(node)
    node.children && list.unshift(...node.children)
  }
}

后序遍歷就略微復(fù)雜一點,我們需要不斷將子樹擴(kuò)展到根節(jié)點前面去乳蓄,(艱難地)執(zhí)行列表遍歷咪橙,遍歷到某個節(jié)點如果它沒有子節(jié)點或者它的子節(jié)點已經(jīng)擴(kuò)展到它前面了,則執(zhí)行訪問操作虚倒,否則擴(kuò)展子節(jié)點到當(dāng)前節(jié)點前面:

function treeForeach (tree, func) {
  let node, list = [...tree], i =  0
  while (node = list[i]) {
    let childCount = node.children ? node.children.length : 0
    if (!childCount || node.children[childCount - 1] === list[i - 1]) {
      func(node)
      i++
    } else {
      list.splice(i, 0, ...node.children)
    }
  }
}

二美侦、列表和樹結(jié)構(gòu)相互轉(zhuǎn)換

1. 列表轉(zhuǎn)為樹

列表結(jié)構(gòu)通常是在節(jié)點信息中給定了父級元素的id,然后通過這個依賴關(guān)系將列表轉(zhuǎn)換為樹形結(jié)構(gòu)魂奥,列表結(jié)構(gòu)是類似于:

let list = [
  {
    id: '1',
    title: '節(jié)點1',
    parentId: '',
  },
  {
    id: '1-1',
    title: '節(jié)點1-1',
    parentId: '1'
  },
  {
    id: '1-2',
    title: '節(jié)點1-2',
   parentId: '1'
  },
  {
    id: '2',
    title: '節(jié)點2',
    parentId: ''
  },
  {
    id: '2-1',
    title: '節(jié)點2-1',
   parentId: '2'
  }
]

列表結(jié)構(gòu)轉(zhuǎn)為樹結(jié)構(gòu)音榜,就是把所有非根節(jié)點放到對應(yīng)父節(jié)點的chilren數(shù)組中,然后把根節(jié)點提取出來:

function listToTree (list) {
  let info = list.reduce((map, node) => (map[node.id] = node, node.children = [], map), {})
  return list.filter(node => {
    info[node.parentId] && info[node.parentId].children.push(node)
    return !node.parentId
  })
}

這里首先通過info建立了id=>node的映射捧弃,因為對象取值的時間復(fù)雜度是O(1)赠叼,這樣在接下來的找尋父元素就不需要再去遍歷一次list了,因為遍歷尋找父元素時間復(fù)雜度是O(n)违霞,并且是在循環(huán)中遍歷嘴办,則總體時間復(fù)雜度會變成O(n^2),而上述實現(xiàn)的總體復(fù)雜度是O(n)买鸽。

2. 樹結(jié)構(gòu)轉(zhuǎn)列表結(jié)構(gòu)

有了遍歷樹結(jié)構(gòu)的經(jīng)驗涧郊,樹結(jié)構(gòu)轉(zhuǎn)為列表結(jié)構(gòu)就很簡單了。不過有時候眼五,我們希望轉(zhuǎn)出來的列表按照目錄展示一樣的順序放到一個列表里的妆艘,并且包含層級信息。使用先序遍歷將樹結(jié)構(gòu)轉(zhuǎn)為列表結(jié)構(gòu)是合適的看幼,直接上代碼:

//遞歸實現(xiàn)
function treeToList (tree, result = [], level = 0) {
  tree.forEach(node => {
    result.push(node)
    node.level = level + 1
    node.children && treeToList(node.children, result, level + 1)
  })
  return result
}

// 循環(huán)實現(xiàn)
function treeToList (tree) {
  let node, result = tree.map(node => (node.level = 1, node))
  for (let i = 0; i < result.length; i++) {
    if (!result[i].children) continue
    let list = result[i].children.map(node => (node.level = result[i].level + 1, node))
    result.splice(i+1, 0, ...list)
  }
  return result
}

三批旺、樹結(jié)構(gòu)篩選

樹結(jié)構(gòu)過濾即保留某些符合條件的節(jié)點,剪裁掉其它節(jié)點诵姜。一個節(jié)點是否保留在過濾后的樹結(jié)構(gòu)中汽煮,取決于它以及后代節(jié)點中是否有符合條件的節(jié)點。可以傳入一個函數(shù)描述符合條件的節(jié)點:

function treeFilter (tree, func) {
  // 使用map復(fù)制一下節(jié)點暇赤,避免修改到原樹
  return tree.map(node => ({ ...node })).filter(node => {
    node.children = node.children && treeFilter(node.children, func)
    return func(node) || (node.children && node.children.length)
  })
}

四心例、樹結(jié)構(gòu)查找

1. 查找節(jié)點

查找節(jié)點其實就是一個遍歷的過程,遍歷到滿足條件的節(jié)點則返回鞋囊,遍歷完成未找到則返回null止后。類似數(shù)組的find方法,傳入一個函數(shù)用于判斷節(jié)點是否符合條件溜腐,代碼如下:

function treeFind (tree, func) {
  for (const data of tree) {
    if (func(data)) return data
    if (data.children) {
      const res = treeFind(data.children, func)
      if (res) return res
    }
  }
  return null
}
2. 查找節(jié)點路徑

略微復(fù)雜一點坯门,因為不知道符合條件的節(jié)點在哪個子樹,要用到回溯法的思想逗扒。查找路徑要使用先序遍歷,維護(hù)一個隊列存儲路徑上每個節(jié)點的id欠橘,假設(shè)節(jié)點就在當(dāng)前分支矩肩,如果當(dāng)前分支查不到,則回溯肃续。

function treeFindPath (tree, func, path = []) {
  if (!tree) return []
  for (const data of tree) {
    path.push(data.id)
    if (func(data)) return path
    if (data.children) {
      const findChildren = treeFindPath(data.children, func, path)
      if (findChildren.length) return findChildren
    }
    path.pop()
  }
  return []
}

用上面的樹結(jié)構(gòu)測試:

let result = treeFindPath(tree, node => node.id === '2-1')
console.log(result)

輸出:

["2","2-1"]
3. 查找多條節(jié)點路徑

思路與查找節(jié)點路徑相似黍檩,不過代碼卻更加簡單:

function treeFindPath (tree, func, path = [], result = []) {
  for (const data of tree) {
    path.push(data.id)
    func(data) && result.push([...path])
    data.children && treeFindPath(data.children, func, path, result)
    path.pop()
  }
  return result
}

參考文獻(xiàn):JS樹結(jié)構(gòu)操作:查找、遍歷始锚、篩選刽酱、樹結(jié)構(gòu)和列表結(jié)構(gòu)相互轉(zhuǎn)換
對于樹結(jié)構(gòu)的操作,其實遞歸是最基礎(chǔ)瞧捌,也是最容易理解的棵里。遞歸本身就是循環(huán)的思想,所以可以用循環(huán)來改寫遞歸姐呐。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末殿怜,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子曙砂,更是在濱河造成了極大的恐慌头谜,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,826評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件鸠澈,死亡現(xiàn)場離奇詭異柱告,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)笑陈,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,968評論 3 395
  • 文/潘曉璐 我一進(jìn)店門际度,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人涵妥,你說我怎么就攤上這事甲脏。” “怎么了?”我有些...
    開封第一講書人閱讀 164,234評論 0 354
  • 文/不壞的土叔 我叫張陵块请,是天一觀的道長娜氏。 經(jīng)常有香客問我,道長墩新,這世上最難降的妖魔是什么贸弥? 我笑而不...
    開封第一講書人閱讀 58,562評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮海渊,結(jié)果婚禮上绵疲,老公的妹妹穿的比我還像新娘。我一直安慰自己臣疑,他們只是感情好盔憨,可當(dāng)我...
    茶點故事閱讀 67,611評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著讯沈,像睡著了一般郁岩。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上缺狠,一...
    開封第一講書人閱讀 51,482評論 1 302
  • 那天问慎,我揣著相機(jī)與錄音,去河邊找鬼挤茄。 笑死如叼,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的穷劈。 我是一名探鬼主播笼恰,決...
    沈念sama閱讀 40,271評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼歇终!你這毒婦竟也來了挖腰?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,166評論 0 276
  • 序言:老撾萬榮一對情侶失蹤练湿,失蹤者是張志新(化名)和其女友劉穎猴仑,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體肥哎,經(jīng)...
    沈念sama閱讀 45,608評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡辽俗,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,814評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了篡诽。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片崖飘。...
    茶點故事閱讀 39,926評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖杈女,靈堂內(nèi)的尸體忽然破棺而出朱浴,到底是詐尸還是另有隱情吊圾,我是刑警寧澤,帶...
    沈念sama閱讀 35,644評論 5 346
  • 正文 年R本政府宣布翰蠢,位于F島的核電站项乒,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏梁沧。R本人自食惡果不足惜檀何,卻給世界環(huán)境...
    茶點故事閱讀 41,249評論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望廷支。 院中可真熱鬧频鉴,春花似錦、人聲如沸恋拍。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,866評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽施敢。三九已至周荐,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間悯姊,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,991評論 1 269
  • 我被黑心中介騙來泰國打工贩毕, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留悯许,地道東北人。 一個月前我還...
    沈念sama閱讀 48,063評論 3 370
  • 正文 我出身青樓辉阶,卻偏偏與公主長得像先壕,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子谆甜,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,871評論 2 354

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