js 將線性數(shù)據(jù)轉(zhuǎn)為樹形的示例代碼
在日常開發(fā)工作中,我們經(jīng)常碰到將線性的數(shù)據(jù)轉(zhuǎn)換成樹的需求,今天給大家分享一個(gè)簡(jiǎn)單的轉(zhuǎn)換算法。
數(shù)據(jù)結(jié)構(gòu)
下面是我們轉(zhuǎn)換前的數(shù)據(jù):
[
{
"id":1,
"parent_id":0,
"name":"四川省"
},
{
"id":2,
"parent_id":0,
"name":"廣東省"
},
{
"id":3,
"parent_id":0,
"name":"江西省"
},
{
"id":5,
"parent_id":1,
"name":"成都市"
},
{
"id":6,
"parent_id":5,
"name":"錦江區(qū)"
},
{
"id":7,
"parent_id":6,
"name":"九眼橋"
},
{
"id":8,
"parent_id":6,
"name":"蘭桂坊"
},
{
"id":9,
"parent_id":2,
"name":"東莞市"
},
{
"id":10,
"parent_id":9,
"name":"長(zhǎng)安鎮(zhèn)"
},
{
"id":11,
"parent_id":3,
"name":"南昌市"
}
]
我們轉(zhuǎn)換后的結(jié)果是:
[
{
"id":1,
"parent_id":0,
"name":"四川省",
"children":[
{
"id":5,
"parent_id":1,
"name":"成都市",
"children":[
{
"id":6,
"parent_id":5,
"name":"錦江區(qū)",
"children":[
{
"id":7,
"parent_id":6,
"name":"九眼橋"
},
{
"id":8,
"parent_id":6,
"name":"蘭桂坊"
}
]
}
]
}
]
},
{
"id":2,
"parent_id":0,
"name":"廣東省",
"children":[
{
"id":9,
"parent_id":2,
"name":"東莞市",
"children":[
{
"id":10,
"parent_id":9,
"name":"長(zhǎng)安鎮(zhèn)"
}
]
}
]
},
{
"id":3,
"parent_id":0,
"name":"江西省",
"children":[
{
"id":11,
"parent_id":3,
"name":"南昌市"
}
]
}
]
實(shí)現(xiàn)代碼
let array = [
{
id: 1,
parent_id: 0,
name: "四川省"
},
{
id: 2,
parent_id: 0,
name: "廣東省"
},
{
id: 3,
parent_id: 0,
name: "江西省"
},
{
id: 5,
parent_id: 1,
name: "成都市"
},
{
id: 6,
parent_id: 5,
name: "錦江區(qū)"
},
{
id: 7,
parent_id: 6,
name: "九眼橋"
},
{
id: 8,
parent_id: 6,
name: "蘭桂坊"
},
{
id: 9,
parent_id: 2,
name: "東莞市"
},
{
id: 10,
parent_id: 9,
name: "長(zhǎng)安鎮(zhèn)"
},
{
id: 11,
parent_id: 3,
name: "南昌市"
}
]
function listToTree(list) {
let map = {};
list.forEach(item => {
if (! map[item.id]) {
map[item.id] = item;
}
});
list.forEach(item => {
if (item.parent_id !== 0) {
map[item.parent_id].children ? map[item.parent_id].children.push(item) : map[item.parent_id].children = [item];
}
});
return list.filter(item => {
if (item.parent_id === 0) {
return item;
}
})
}
console.log(listToTree(array));
分析
這段代碼的核心就在 listToTree 方法中,這個(gè)方法分為了三個(gè)部分:
第一部分
第一部分先將數(shù)組中的所有元素都復(fù)制到 map 中(注意:這里是引用復(fù)制哦,這個(gè)細(xì)節(jié)很重要)。
第二部分
執(zhí)行第二次遍歷前的 map:
// map
{
...,
"3":{
"id":3,
"parent_id":0,
"name":"江西省"
},
...
}
然后這個(gè)時(shí)候遍歷 parent_id 不等于 0 的元素:
[
...,
{
id: 11,
parent_id: 3,
name: "南昌市"
},
...
]
然后發(fā)現(xiàn)南昌市有 parent_id,我們?cè)俳o map[item.parent_id] 設(shè)置子元素,通過南昌市的 parent_id 可以推導(dǎo)出:
map["3"].children ? map["3"].children.push(item) : map[3].children = [item];
上面的代碼判斷了是否存在 children,如果不存在則直接給它賦值,否則將值 push 到 children 中。
執(zhí)行完第二步后,我們已經(jīng)把子節(jié)點(diǎn)添加到了它的父節(jié)點(diǎn)上,但是我們并沒有刪除掉之前的子節(jié)點(diǎn)。所以第三部就是對(duì)數(shù)據(jù)進(jìn)行過濾,只要父節(jié)點(diǎn)即可。

總結(jié)
需要注意的是,我們一直都是對(duì) map 進(jìn)行操作的,但是結(jié)果怎么到了 list 上呢,這就是上面提到的引用復(fù)制。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
firefox和IE系列的相關(guān)區(qū)別整理 以備后用
firefox和IE系列的相關(guān)區(qū)別整理,整理相對(duì)來說還可以,但對(duì)于個(gè)別細(xì)節(jié)的處理不夠完善。具體的可以參考腳本*之家以前發(fā)布的文章。2009-12-12
JavaScript實(shí)現(xiàn)單擊網(wǎng)頁任意位置打開新窗口與關(guān)閉窗口的方法
這篇文章主要介紹了JavaScript實(shí)現(xiàn)單擊網(wǎng)頁任意位置打開新窗口與關(guān)閉窗口的方法,涉及javascript窗口的相關(guān)操作函數(shù)與使用技巧,需要的朋友可以參考下2017-09-09
如何通過JavaScript來實(shí)現(xiàn)頁面間數(shù)據(jù)傳遞
這篇文章主要給大家介紹了關(guān)于如何通過JavaScript來實(shí)現(xiàn)頁面間數(shù)據(jù)傳遞的相關(guān)資料,在前端開發(fā)中我們常常需要從一個(gè)跳到另一個(gè)頁面,并且將當(dāng)前頁面的數(shù)據(jù)傳遞過去,需要的朋友可以參考下2023-11-11
ant-design-pro?的EditableProTable表格驗(yàn)證調(diào)用的實(shí)現(xiàn)代碼
這篇文章主要介紹了ant-design-pro?的EditableProTable表格驗(yàn)證調(diào)用,這里的需求是點(diǎn)擊外部的保存要對(duì)整個(gè)表單進(jìn)行驗(yàn)證,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2022-06-06
前端實(shí)現(xiàn)(excel)xlsx文件預(yù)覽的詳細(xì)步驟
excel的預(yù)覽庫有不少,也都很強(qiáng)大,但是能很簡(jiǎn)單實(shí)現(xiàn),下面這篇文章主要給大家介紹了關(guān)于前端實(shí)現(xiàn)(excel)xlsx文件預(yù)覽的詳細(xì)步驟,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-03-03
JavaScript寫的一個(gè)DIV 彈出網(wǎng)頁對(duì)話框
自己整理得一個(gè)JavaScript寫的一個(gè)DIV 彈出網(wǎng)頁對(duì)話框2009-08-08
基于JavaScript實(shí)現(xiàn)驗(yàn)證碼功能
這篇文章主要介紹了基于JavaScript實(shí)現(xiàn)驗(yàn)證碼功能的相關(guān)資料2017-04-04
JS面向?qū)ο缶幊獭狤S6 中class的繼承用法詳解
這篇文章主要介紹了JS面向?qū)ο缶幊獭狤S6 中class的繼承用法,結(jié)合實(shí)例形式詳細(xì)分析了ES6中class繼承的相關(guān)操作技巧與使用注意事項(xiàng),需要的朋友可以參考下2020-03-03

