詳解Java遞歸實(shí)現(xiàn)樹(shù)形結(jié)構(gòu)的兩種方式
0、引言
在開(kāi)發(fā)的過(guò)程中,很多業(yè)務(wù)場(chǎng)景需要一個(gè)樹(shù)形結(jié)構(gòu)的結(jié)果集進(jìn)行前端展示,也可以理解為是一個(gè)無(wú)限父子結(jié)構(gòu),常見(jiàn)的有報(bào)表指標(biāo)結(jié)構(gòu)、菜單結(jié)構(gòu)等。Java中遞歸實(shí)現(xiàn)樹(shù)形結(jié)構(gòu)的兩種常見(jiàn)方式如下:
- Java7及以下純Java遞歸實(shí)現(xiàn)
- Java8及以上借助lamda表達(dá)式實(shí)現(xiàn)
1、數(shù)據(jù)準(zhǔn)備
Java實(shí)體類NodePO對(duì)應(yīng)數(shù)據(jù)庫(kù)表
package com.wbs.pojo; import lombok.Data; import lombok.NoArgsConstructor; import java.util.List; @Data @NoArgsConstructor public class NodePO { /** * 當(dāng)前節(jié)點(diǎn)id */ private String id; /** * 當(dāng)前節(jié)點(diǎn)名稱 */ private String name; /** * 父級(jí)節(jié)點(diǎn)id */ private String parentId; /** * 當(dāng)前節(jié)點(diǎn)序號(hào) */ private String orderNo; /** * 子集節(jié)點(diǎn) */ private List<NodePO> children; /** * 構(gòu)造函數(shù) * @param id * @param name * @param parentId * @param orderNo */ public NodePO(String id,String name,String parentId,String orderNo){ this.id = id; this.name = name; this.parentId = parentId; this.orderNo = orderNo; } }
? 自己造一些數(shù)據(jù)模擬從數(shù)據(jù)庫(kù)中查詢出來(lái)的數(shù)據(jù):
static final List<NodePO> nodePOs = Arrays.asList( new NodePO("1","一級(jí)節(jié)點(diǎn)1",null,"_0001"), new NodePO("2","二級(jí)節(jié)點(diǎn)1.1","1","_0002"), new NodePO("3","二級(jí)節(jié)點(diǎn)1.2","1","_0003"), new NodePO("4","一級(jí)節(jié)點(diǎn)2",null,"_0004"), new NodePO("5","二級(jí)節(jié)點(diǎn)2.1","4","_0005"), new NodePO("6","二級(jí)節(jié)點(diǎn)2.2","4","_0006"), new NodePO("7","三級(jí)節(jié)點(diǎn)2.2.1","6","_0007"), new NodePO("8","一級(jí)節(jié)點(diǎn)3",null,"_0008"), new NodePO("9","二級(jí)節(jié)點(diǎn)3.1","8","_0009"), new NodePO("10","三級(jí)節(jié)點(diǎn)3.1.1","9","_0010"), new NodePO("11","四級(jí)節(jié)點(diǎn)3.1.1.1","10","_0011"), new NodePO("12","五級(jí)節(jié)點(diǎn)3.1.1.1.1","11","_0012") );
2、類型轉(zhuǎn)化
從開(kāi)發(fā)的過(guò)程中發(fā)現(xiàn)直接操作實(shí)體類集合,專門指定某一個(gè)實(shí)體類封裝的方法是不具有普適性的,所以將實(shí)體類集合統(tǒng)一轉(zhuǎn)化為Map集合,操作方便,具有一定的普適性:
List<Map<String, Object>> mapList = BeanMapUtils.listBeanToListMap(jsonObject);
BeanMapUtils自己簡(jiǎn)單封裝一個(gè)工具類(不懼普適性勿噴):
package com.wbs.util; import com.alibaba.fastjson.JSON; import com.alibaba.fastjson.JSONObject; import com.google.common.collect.Lists; import com.google.common.collect.Maps; import lombok.SneakyThrows; import org.springframework.cglib.beans.BeanMap; import java.util.*; import java.util.function.Function; import java.util.stream.Collectors; /** * @author 一宿君 * @version Id: BeanMapUtils.java, v 0.1 Administrator Exp $$ * @date 2022-10-13 14:24:20 * @desc java實(shí)體類和map相互轉(zhuǎn)換工具類 */ public class BeanMapUtils { /** * 將實(shí)體類對(duì)象屬性轉(zhuǎn)化為map對(duì)象 * @param t * @param <T> * @return */ public static <T> Map<String, Object> beanToMap(T t) { Map<String, Object> map = new HashMap<>(); if (t != null) { if (t instanceof JSONObject){ return (JSONObject)t; } BeanMap beanMap = BeanMap.create(t); for (Object key : beanMap.keySet()) { map.put(key.toString(), beanMap.get(key)); } } return map; } /** * 將map對(duì)象中轉(zhuǎn)化為實(shí)體類對(duì)象 * @param map * @param clazz * @param <T> * @return * @throws Exception */ public static <T> T mapToBean(Map<String, Object> map,Class<T> clazz) throws Exception { T bean = clazz.newInstance(); if (bean instanceof JSONObject){ JSONObject jsonObject = (JSONObject)bean; Set<Map.Entry<String, Object>> entries = map.entrySet(); for (Map.Entry<String, Object> entry : entries) { jsonObject.put(entry.getKey(),entry.getValue()); } return (T)jsonObject; } BeanMap beanMap = BeanMap.create(bean); beanMap.putAll(map); return bean; } /** * 通過(guò)lambda表達(dá)式將List<JavaBean>轉(zhuǎn)化為L(zhǎng)ist<Map<String, Object>> * @param objList * @param <T> * @return */ public static <T> List<Map<String, Object>> listBeanToListMap(List<T> objList) { return objList.stream().map(new Function<T, Map<String, Object>>() { @Override public Map<String, Object> apply(T t) { Map<String,Object> map = Maps.newHashMap(); if (t instanceof JSONObject){ return (JSONObject)t; } BeanMap beanMap = BeanMap.create(t); for (Object key : beanMap.keySet()) { map.put(key.toString(), beanMap.get(key)); } return map; } }).collect(Collectors.toList()); } /** * 通過(guò)lambda表達(dá)式將List<Map<String, Object>>轉(zhuǎn)化為L(zhǎng)ist<JavaBean> * @param mapList * @param <T> * @return */ public static <T> List<T> listMapToListBean(List<Map<String,Object>> mapList,Class<T> clazz) { return mapList.stream().map(new Function<Map<String, Object>,T>() { @SneakyThrows @Override public T apply(Map<String, Object> map) { T t = clazz.newInstance(); if (t instanceof JSONObject){ return (T)map; } BeanMap beanMap = BeanMap.create(t); beanMap.putAll(map); return t; } }).collect(Collectors.toList()); } }
其中org.springframework.cglib.beans.BeanMap;
是org.springframework:spring-core
依賴下的工具包,spring-core
核心依賴只要導(dǎo)入spring-boot-starter
依賴即可
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter</artifactId> <version>2.2.0.RELEASE</version> </dependency>
3、遞歸實(shí)現(xiàn)方法
3.1、Java7及以下純Java遞歸實(shí)現(xiàn)
既然是Java7及以下實(shí)現(xiàn)方式,那排序也用最原始的冒泡排序:
/** * 冒泡排序,小的在前,大的在后 * @param list * @return */ public static List<Map<String, Object>> sortJava7Map(List<Map<String, Object>> list){ if(CollectionUtils.isEmpty(list)){ return Lists.newArrayList(); } boolean flag; int size = list.size(); for (int i = 0; i < size - 1; i++) { flag = false; for (int j = 1; j < size - i; j++) { Map<String, Object> frontMap = list.get(j - 1); Map<String, Object> afterMap = list.get(j); if (String.valueOf(frontMap.get("orderNo")).compareTo(String.valueOf(afterMap.get("orderNo"))) > 0){ list.set(j - 1,afterMap); list.set(j,frontMap); flag = true; } } //如果沒(méi)有發(fā)生位置互換,則退出循環(huán) if (!flag){ break; } } return list; }
給定一個(gè)節(jié)點(diǎn),獲取它的所有子節(jié)點(diǎn):
/** * Java7及以下版本獲取子節(jié)點(diǎn)的方式 * @param parentNode * @param allList * @return */ public static List<Map<String, Object>> getJava7Children(Map<String,Object> parentNode,List<Map<String, Object>> allList){ //存放當(dāng)前節(jié)點(diǎn)的直系子節(jié)點(diǎn) List<Map<String, Object>> curNodeChildrenList = Lists.newArrayList(); //存放直系子節(jié)點(diǎn)以外的節(jié)點(diǎn) List<Map<String, Object>> otherNodeList = Lists.newArrayList(); Object pId = parentNode.get("id"); for (Map<String, Object> map : allList) { Object curPId = map.get("parentId"); if (ObjectUtils.isNotEmpty(curPId) && Objects.equals(pId,curPId)){ curNodeChildrenList.add(map); }else { otherNodeList.add(map); } } if (curNodeChildrenList.isEmpty()){ return curNodeChildrenList; } //每一層級(jí)都進(jìn)行排序 curNodeChildrenList = sortJava7Map(curNodeChildrenList); //迭代直系子節(jié)點(diǎn)再獲取子節(jié)點(diǎn) for (Map<String, Object> map : curNodeChildrenList) { map.put("children",getJava7Children(map,otherNodeList)); } return curNodeChildrenList; }
給出一個(gè)結(jié)果集,構(gòu)建樹(shù)形結(jié)果集:
/** * 使用Java7的方式獲取樹(shù)形結(jié)構(gòu) * @param allList * @return */ public static List<Map<String, Object>> getJava7ResultTree(List<Map<String, Object>> allList){ //存放所有的一級(jí)節(jié)點(diǎn) List<Map<String, Object>> oneLevelNodeList = Lists.newArrayList(); for (Map<String, Object> map : allList) { if (ObjectUtils.isEmpty(map.get("parentId"))){ map.put("children",getJava7Children(map,allList)); oneLevelNodeList.add(map); } } return sortJava8Map(oneLevelNodeList); }
獲取樹(shù)形結(jié)構(gòu):
//轉(zhuǎn)化為Map集合 List<Map<String, Object>> mapList = BeanMapUtils.listBeanToListMap(nodePOs); //獲取樹(shù)形結(jié)構(gòu) List<Map<String, Object>> java7ResultTree = getJava7ResultTree(mapList); //打印輸出 System.out.println(JSON.toJSONString(java7ResultTree));
打印結(jié)果:
[{"orderNo":"_0001","children":[{"orderNo":"_0002","children":[],"name":"二級(jí)節(jié)點(diǎn)1.1","id":"2","parentId":"1"},{"orderNo":"_0003","children":[],"name":"二級(jí)節(jié)點(diǎn)1.2","id":"3","parentId":"1"}],"name":"一級(jí)節(jié)點(diǎn)1","id":"1"},{"orderNo":"_0004","children":[{"orderNo":"_0005","children":[],"name":"二級(jí)節(jié)點(diǎn)2.1","id":"5","parentId":"4"},{"orderNo":"_0006","children":[{"orderNo":"_0007","children":[],"name":"三級(jí)節(jié)點(diǎn)2.2.1","id":"7","parentId":"6"}],"name":"二級(jí)節(jié)點(diǎn)2.2","id":"6","parentId":"4"}],"name":"一級(jí)節(jié)點(diǎn)2","id":"4"},{"orderNo":"_0008","children":[{"orderNo":"_0009","children":[{"orderNo":"_0010","children":[{"orderNo":"_0011","children":[{"orderNo":"_0012","children":[],"name":"五級(jí)節(jié)點(diǎn)3.1.1.1.1","id":"12","parentId":"11"}],"name":"四級(jí)節(jié)點(diǎn)3.1.1.1","id":"11","parentId":"10"}],"name":"三級(jí)節(jié)點(diǎn)3.1.1","id":"10","parentId":"9"}],"name":"二級(jí)節(jié)點(diǎn)3.1","id":"9","parentId":"8"}],"name":"一級(jí)節(jié)點(diǎn)3","id":"8"}]
樹(shù)形結(jié)構(gòu)搞定!
3.2、Java8及以上借助lamda表達(dá)式實(shí)現(xiàn)
Java7的方式雖然實(shí)現(xiàn)了樹(shù)形結(jié)構(gòu),但是有一定的缺點(diǎn),比如:代碼量比較大,邏輯相對(duì)較復(fù)雜,那Java8是如何簡(jiǎn)化,如下所示:
既然Java8有l(wèi)amda表達(dá)式,那代碼我們能省就省,先看排序,一行代碼搞定:
/** * 根據(jù)orderNo排序樹(shù)形結(jié)構(gòu)的每一個(gè)層級(jí) * @param list * @return */ public static List<Map<String, Object>> sortJava8Map(List<Map<String, Object>> list){ if(CollectionUtils.isEmpty(list)){ return Lists.newArrayList(); } //關(guān)鍵之處,一行代碼搞定 list.sort(Comparator.comparing(m -> String.valueOf(m.get("orderNo")))); return list; }
給定一個(gè)節(jié)點(diǎn),獲取它的所有子節(jié)點(diǎn):
釋義:
filter: 過(guò)濾,相當(dāng)于for循環(huán),再if條件判斷。
peek: 給定一個(gè)節(jié)點(diǎn),往它的children塞子節(jié)點(diǎn)。
/** * 根據(jù)父級(jí)節(jié)點(diǎn)獲取所有的子集節(jié)點(diǎn) * @param parentNode * @param allList * @return */ public static List<Map<String, Object>> getJava8Children(Map<String,Object> parentNode, List<Map<String, Object>> allList){ return allList.stream() .filter(curNode -> ObjectUtils.isNotEmpty(curNode.get("parentId")) && Objects.equals(curNode.get("parentId"),parentNode.get("id"))) .peek(m -> m.put("children", getJava8Children(m,allList))).collect(Collectors.toList()); }
給出一個(gè)結(jié)果集,構(gòu)建樹(shù)形結(jié)果集:
/** * 獲取樹(shù)形結(jié)構(gòu) * @param mapList * @return treeList 樹(shù)形結(jié)果集 */ public static List<Map<String, Object>> getJava8ResultTree(List<Map<String, Object>> mapList){ if (CollectionUtils.isEmpty(mapList)){ return Lists.newArrayList(); } //filter過(guò)濾出所有的一級(jí)節(jié)點(diǎn) return mapList.stream().filter(m -> Objects.equals(m.get("parentId"), null) || Objects.equals(m.get("parentId"), "")) .peek(m -> m.put("children", sortJava8Map(getJava8Children(m, mapList)))).collect(Collectors.toList()); }
獲取樹(shù)形結(jié)構(gòu):
//轉(zhuǎn)化為Map集合 List<Map<String, Object>> mapList = BeanMapUtils.listBeanToListMap(nodePOs); //獲取樹(shù)形結(jié)構(gòu) List<Map<String, Object>> java8ResultTree = getJava8ResultTree(mapList); //打印輸出 System.out.println(JSON.toJSONString(java8ResultTree));
打印結(jié)果:
[{"orderNo":"_0001","children":[{"orderNo":"_0002","children":[],"name":"二級(jí)節(jié)點(diǎn)1.1","id":"2","parentId":"1"},{"orderNo":"_0003","children":[],"name":"二級(jí)節(jié)點(diǎn)1.2","id":"3","parentId":"1"}],"name":"一級(jí)節(jié)點(diǎn)1","id":"1"},{"orderNo":"_0004","children":[{"orderNo":"_0005","children":[],"name":"二級(jí)節(jié)點(diǎn)2.1","id":"5","parentId":"4"},{"orderNo":"_0006","children":[{"orderNo":"_0007","children":[],"name":"三級(jí)節(jié)點(diǎn)2.2.1","id":"7","parentId":"6"}],"name":"二級(jí)節(jié)點(diǎn)2.2","id":"6","parentId":"4"}],"name":"一級(jí)節(jié)點(diǎn)2","id":"4"},{"orderNo":"_0008","children":[{"orderNo":"_0009","children":[{"orderNo":"_0010","children":[{"orderNo":"_0011","children":[{"orderNo":"_0012","children":[],"name":"五級(jí)節(jié)點(diǎn)3.1.1.1.1","id":"12","parentId":"11"}],"name":"四級(jí)節(jié)點(diǎn)3.1.1.1","id":"11","parentId":"10"}],"name":"三級(jí)節(jié)點(diǎn)3.1.1","id":"10","parentId":"9"}],"name":"二級(jí)節(jié)點(diǎn)3.1","id":"9","parentId":"8"}],"name":"一級(jí)節(jié)點(diǎn)3","id":"8"}]
樹(shù)形結(jié)構(gòu)搞定!兩種實(shí)現(xiàn)方式對(duì)比一下,你就說(shuō)Java8的方式哇塞不哇塞?。?!
到此這篇關(guān)于Java遞歸實(shí)現(xiàn)樹(shù)形結(jié)構(gòu)的兩種方式的文章就介紹到這了,更多相關(guān)Java遞歸實(shí)現(xiàn)樹(shù)形結(jié)構(gòu)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- 一文帶你徹底了解Java8中的Lambda,函數(shù)式接口和Stream
- Java的Lambda表達(dá)式和Stream流的作用以及示例
- Java分析Lambda表達(dá)式Stream流合并分組內(nèi)對(duì)象數(shù)據(jù)合并
- Java中的lambda和stream實(shí)現(xiàn)排序
- Java詳細(xì)分析Lambda表達(dá)式與Stream流的使用方法
- 吊打Java面試官之Lambda表達(dá)式 Stream API
- Java8中Lambda表達(dá)式使用和Stream API詳解
- Java實(shí)現(xiàn)樹(shù)形結(jié)構(gòu)的示例代碼
- Java樹(shù)形結(jié)構(gòu)數(shù)據(jù)生成導(dǎo)出excel文件方法記錄
- Java使用 Stream 流和 Lambda 組裝復(fù)雜父子樹(shù)形結(jié)構(gòu)
相關(guān)文章
java實(shí)現(xiàn)后臺(tái)處理base64圖片還原為文件
這篇文章主要介紹了java實(shí)現(xiàn)后臺(tái)處理base64圖片還原為文件,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02java實(shí)現(xiàn)ThreadLocal線程局部變量的實(shí)現(xiàn)
本文主要介紹了java實(shí)現(xiàn)ThreadLocal線程局部變量的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-07-07Java?DirectByteBuffer堆外內(nèi)存回收詳解
這篇文章主要為大家詳細(xì)介紹了Java中發(fā)DirectByteBuffer堆外內(nèi)存回收,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,需要的可以參考一下2022-10-10idea已經(jīng)提交到遠(yuǎn)程分支,但需要本地和遠(yuǎn)程都回退到某一版本問(wèn)題
這篇文章主要介紹了idea已經(jīng)提交到遠(yuǎn)程分支,但需要本地和遠(yuǎn)程都回退到某一版本問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-11-11MyBatis-plus報(bào)錯(cuò)Property ‘sqlSessionFactory‘ or 
這篇文章主要給大家介紹了MyBatis-plus 報(bào)錯(cuò) Property ‘sqlSessionFactory‘ or ‘sqlSessionTemplate‘ are required的兩種解決方法,如果遇到相同問(wèn)題的朋友可以參考借鑒一下2023-12-12