Java實(shí)現(xiàn)樹狀結(jié)構(gòu)解析

最近的工作有遇到此需求,如權(quán)限樹狀結(jié)構(gòu)。因?yàn)閿?shù)據(jù)庫(kù)采用的MySQL,其在語義層面對(duì)于樹狀結(jié)構(gòu)的查詢支持偏弱早芭。查詢了諸多資料,個(gè)人感覺靠譜的不多遂萌生了自己寫一個(gè)的想法诅蝶。話不多說啊退个,直接貼代碼:

package tree;

import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.stream.Collectors;

import org.apache.commons.lang3.builder.EqualsBuilder;

import com.alibaba.fastjson.JSON;

import lombok.Getter;
import lombok.Setter;
import lombok.ToString;

/**
 * 樹形結(jié)構(gòu)解析
 * 
 * @author gewx
 **/
public final class Tree {

    /**
     * 解析數(shù)據(jù)結(jié)構(gòu)
     * 
     * @author gewx
     * @param nodeList 數(shù)據(jù)節(jié)點(diǎn)集合
     * @return 解析完成后的樹狀結(jié)果
     **/
    public static List<Node> parse(List<Node> nodeList) {
        List<Node> resultList = new ArrayList<>(64);
        nodeList.stream().filter(val -> !hasChild(nodeList, val.getMenuId())).collect(Collectors.toList())
                .forEach(val -> {
                    reverseRecursion(val, nodeList, resultList);
                });

        Set<Node> set = new HashSet<>(32);
        set.addAll(resultList);

        resultList.clear();
        set.stream().forEach(val -> {
            resultList.add(val);
            recursion(val, nodeList);
        });

        return resultList;
    }

    /**
     * 取出某個(gè)節(jié)點(diǎn)直至末尾葉子節(jié)點(diǎn)的數(shù)據(jù)
     * 
     * @author gewx
     * @param node     葉子節(jié)點(diǎn)
     * @param nodeList 數(shù)據(jù)節(jié)點(diǎn)集合
     * @return 解析完成后的樹狀結(jié)果
     **/
    public static void getNodeJson(Node node, List<Node> nodeList) {
        recursion(node, nodeList);
    }

    /**
     * 倒序遞歸檢索所有父節(jié)點(diǎn),由下往上直到所有根節(jié)點(diǎn)
     * 
     * @author gewx
     * @param node       末尾葉子節(jié)點(diǎn)
     * @param nodeList   數(shù)據(jù)節(jié)點(diǎn)集合
     * @param resultList 根節(jié)點(diǎn)集合
     * @return void
     **/
    private static void reverseRecursion(Node node, List<Node> nodeList, List<Node> resultList) {
        List<Node> parentNodeList = nodeList.stream().filter(val -> val.getMenuId().equals(node.getParentId()))
                .collect(Collectors.toList());
        if (parentNodeList.size() != 0) {
            Node parentNode = parentNodeList.get(0);
            reverseRecursion(parentNode, nodeList, resultList);
        } else {
            resultList.add(node);
        }
    }

    /**
     * 遞歸檢索所有子節(jié)點(diǎn)
     * 
     * @author gewx
     * @param node     根節(jié)點(diǎn)
     * @param nodeList 數(shù)據(jù)節(jié)點(diǎn)集合
     * @return void
     **/
    private static void recursion(Node node, List<Node> nodeList) {
        List<Node> childNodeList = nodeList.stream().filter(val -> val.getParentId().equals(node.getMenuId()))
                .collect(Collectors.toList());
        if (childNodeList.size() != 0) {
            node.setChildren(childNodeList);
            childNodeList.stream().forEach(val -> {
                recursion(val, nodeList);
            });
        }
    }

    /**
     * 是否還存在父級(jí)元素,找出末尾葉子節(jié)點(diǎn)
     * 
     * @author gewx
     * 
     * @param node   葉子節(jié)點(diǎn)
     * @param menuId 節(jié)點(diǎn)標(biāo)記
     * @return boolean true 存在, false 不存在
     **/
    private static boolean hasChild(List<Node> node, String menuId) {
        return node.stream().anyMatch(val -> val.getParentId().equals(menuId));
    }

    /**
     * 葉子節(jié)點(diǎn)數(shù)據(jù)結(jié)構(gòu),可與庫(kù)表做一一映射僅需要保持menuId與parentId映射關(guān)系即可
     **/
    @Getter
    @Setter
    @ToString
    public static class Node {

        /**
         * 菜單Id
         **/
        private String menuId;

        /**
         * 菜單名稱
         **/
        private String menuName;

        /**
         * 菜單類型
         **/
        private String menuType;

        /**
         * 菜單編碼
         **/
        private String menuCode;

        /**
         * 菜單URL
         **/
        private String url;

        /**
         * 菜單父級(jí)Id
         **/
        public String parentId;

        /**
         * 排序
         **/
        private Integer sortNum;

        /**
         * 子菜單
         **/
        private List<Node> children = new ArrayList<>();

        public Node(String menuId, String menuName, String menuType, String menuCode, String url, String parentId,
                Integer sortNum) {
            this.menuId = menuId;
            this.menuName = menuName;
            this.menuType = menuType;
            this.menuCode = menuCode;
            this.url = url;
            this.parentId = parentId;
            this.sortNum = sortNum;
        }

        @Override
        public final boolean equals(Object obj) {
            if (obj == null) {
                return false;
            }

            Node otherObject = (Node) obj;
            if (getClass() != otherObject.getClass()) {
                return false;
            }

            EqualsBuilder builder = new EqualsBuilder();
            builder.append(this.menuId, otherObject.getMenuId());
            return builder.isEquals();
        }

        @Override
        public final int hashCode() {
            return this.menuId.hashCode() * 31;
        }
    }

    public static void main(String[] args) {
        List<Node> array = new ArrayList<>();
        array.add(new Tree.Node("CODE_00", "用戶管理", "M", "0101", "#", "0", 0));
        array.add(new Tree.Node("CODE_01", "管理界面", "M", "0102", "#", "0", 0));
        array.add(new Tree.Node("CODE_02", "權(quán)限管理", "C", "010101", "http://www.baidu.com", "CODE_00", 0));
        array.add(new Tree.Node("CODE_03", "內(nèi)部管理", "C", "010201", "http://www.baidu.com", "CODE_01", 0));

        array.add(new Tree.Node("CODE_04", "內(nèi)部用戶權(quán)限", "C", "010201", "http://www.baidu.com", "CODE_02", 0));
        array.add(new Tree.Node("CODE_05", "外部用戶權(quán)限", "C", "010201", "http://www.baidu.com", "CODE_02", 0));
        array.add(new Tree.Node("CODE_06", "子權(quán)限", "C", "010201", "http://www.baidu.com", "CODE_04", 0));
        array.add(new Tree.Node("CODE_07", "子系統(tǒng)內(nèi)部管理", "C", "010201", "http://www.baidu.com", "CODE_03", 0));

        List<Node> list = Tree.parse(array);
        String val = JSON.toJSONString(list);
        System.out.println(val);

        // 取出當(dāng)下節(jié)點(diǎn)下所有數(shù)據(jù)
        Node node = new Tree.Node("CODE_01x", "系統(tǒng)管理員", "M", "0101", "#", "0", 0);
        Tree.getNodeJson(node, array);
        System.out.println(JSON.toJSONString(node));
    }
}

解析結(jié)果:

[
    {
        "children": [
            {
                "children": [
                    {
                        "children": [],
                        "menuCode": "010201",
                        "menuId": "CODE_07",
                        "menuName": "子系統(tǒng)內(nèi)部管理",
                        "menuType": "C",
                        "parentId": "CODE_03",
                        "sortNum": 0,
                        "url": "http://www.baidu.com"
                    }
                ],
                "menuCode": "010201",
                "menuId": "CODE_03",
                "menuName": "內(nèi)部管理",
                "menuType": "C",
                "parentId": "CODE_01",
                "sortNum": 0,
                "url": "http://www.baidu.com"
            }
        ],
        "menuCode": "0102",
        "menuId": "CODE_01",
        "menuName": "管理界面",
        "menuType": "M",
        "parentId": "0",
        "sortNum": 0,
        "url": "#"
    },
    {
        "children": [
            {
                "children": [
                    {
                        "children": [
                            {
                                "children": [],
                                "menuCode": "010201",
                                "menuId": "CODE_06",
                                "menuName": "子權(quán)限",
                                "menuType": "C",
                                "parentId": "CODE_04",
                                "sortNum": 0,
                                "url": "http://www.baidu.com"
                            }
                        ],
                        "menuCode": "010201",
                        "menuId": "CODE_04",
                        "menuName": "內(nèi)部用戶權(quán)限",
                        "menuType": "C",
                        "parentId": "CODE_02",
                        "sortNum": 0,
                        "url": "http://www.baidu.com"
                    },
                    {
                        "children": [],
                        "menuCode": "010201",
                        "menuId": "CODE_05",
                        "menuName": "外部用戶權(quán)限",
                        "menuType": "C",
                        "parentId": "CODE_02",
                        "sortNum": 0,
                        "url": "http://www.baidu.com"
                    }
                ],
                "menuCode": "010101",
                "menuId": "CODE_02",
                "menuName": "權(quán)限管理",
                "menuType": "C",
                "parentId": "CODE_00",
                "sortNum": 0,
                "url": "http://www.baidu.com"
            }
        ],
        "menuCode": "0101",
        "menuId": "CODE_00",
        "menuName": "用戶管理",
        "menuType": "M",
        "parentId": "0",
        "sortNum": 0,
        "url": "#"
    }
]

支持無限級(jí)聯(lián)樹,僅需要保證數(shù)據(jù)結(jié)構(gòu)中menuId與parentId需要存在級(jí)聯(lián)關(guān)系即可调炬。個(gè)人簡(jiǎn)單測(cè)試過语盈,小伙伴有需要可以再度驗(yàn)證下。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末缰泡,一起剝皮案震驚了整個(gè)濱河市刀荒,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌棘钞,老刑警劉巖缠借,帶你破解...
    沈念sama閱讀 206,968評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異宜猜,居然都是意外死亡泼返,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門姨拥,熙熙樓的掌柜王于貴愁眉苦臉地迎上來绅喉,“玉大人,你說我怎么就攤上這事叫乌〔窆蓿” “怎么了?”我有些...
    開封第一講書人閱讀 153,220評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵憨奸,是天一觀的道長(zhǎng)革屠。 經(jīng)常有香客問我,道長(zhǎng),這世上最難降的妖魔是什么屠阻? 我笑而不...
    開封第一講書人閱讀 55,416評(píng)論 1 279
  • 正文 為了忘掉前任红省,我火速辦了婚禮,結(jié)果婚禮上国觉,老公的妹妹穿的比我還像新娘吧恃。我一直安慰自己,他們只是感情好麻诀,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,425評(píng)論 5 374
  • 文/花漫 我一把揭開白布痕寓。 她就那樣靜靜地躺著,像睡著了一般蝇闭。 火紅的嫁衣襯著肌膚如雪呻率。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,144評(píng)論 1 285
  • 那天呻引,我揣著相機(jī)與錄音礼仗,去河邊找鬼。 笑死逻悠,一個(gè)胖子當(dāng)著我的面吹牛元践,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播童谒,決...
    沈念sama閱讀 38,432評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼单旁,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了饥伊?” 一聲冷哼從身側(cè)響起象浑,我...
    開封第一講書人閱讀 37,088評(píng)論 0 261
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎琅豆,沒想到半個(gè)月后愉豺,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,586評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡趋距,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,028評(píng)論 2 325
  • 正文 我和宋清朗相戀三年粒氧,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片节腐。...
    茶點(diǎn)故事閱讀 38,137評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖摘盆,靈堂內(nèi)的尸體忽然破棺而出翼雀,到底是詐尸還是另有隱情,我是刑警寧澤孩擂,帶...
    沈念sama閱讀 33,783評(píng)論 4 324
  • 正文 年R本政府宣布狼渊,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏狈邑。R本人自食惡果不足惜城须,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,343評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望米苹。 院中可真熱鬧糕伐,春花似錦、人聲如沸蘸嘶。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)训唱。三九已至褥蚯,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間况增,已是汗流浹背赞庶。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評(píng)論 1 262
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留澳骤,地道東北人尘执。 一個(gè)月前我還...
    沈念sama閱讀 45,595評(píng)論 2 355
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像宴凉,于是被迫代替她去往敵國(guó)和親誊锭。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,901評(píng)論 2 345