Java实现树状结构解析

最近的工作有遇到此需求,如权限树状结构。因为数据库采用的MySQL,其在语义层面对于树状结构的查询支持偏弱。查询了诸多资料,个人感觉靠谱的不多遂萌生了自己写一个的想法。话不多说啊,直接贴代码:

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;

/**
 * 树形结构解析
 * 
 * @author gewx
 **/
public final class Tree {

    /**
     * 解析数据结构
     * 
     * @author gewx
     * @param nodeList 数据节点集合
     * @return 解析完成后的树状结果
     **/
    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;
    }

    /**
     * 取出某个节点直至末尾叶子节点的数据
     * 
     * @author gewx
     * @param node     叶子节点
     * @param nodeList 数据节点集合
     * @return 解析完成后的树状结果
     **/
    public static void getNodeJson(Node node, List<Node> nodeList) {
        recursion(node, nodeList);
    }

    /**
     * 倒序递归检索所有父节点,由下往上直到所有根节点
     * 
     * @author gewx
     * @param node       末尾叶子节点
     * @param nodeList   数据节点集合
     * @param resultList 根节点集合
     * @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);
        }
    }

    /**
     * 递归检索所有子节点
     * 
     * @author gewx
     * @param node     根节点
     * @param nodeList 数据节点集合
     * @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);
            });
        }
    }

    /**
     * 是否还存在父级元素,找出末尾叶子节点
     * 
     * @author gewx
     * 
     * @param node   叶子节点
     * @param menuId 节点标记
     * @return boolean true 存在, false 不存在
     **/
    private static boolean hasChild(List<Node> node, String menuId) {
        return node.stream().anyMatch(val -> val.getParentId().equals(menuId));
    }

    /**
     * 叶子节点数据结构,可与库表做一一映射仅需要保持menuId与parentId映射关系即可
     **/
    @Getter
    @Setter
    @ToString
    public static class Node {

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

        /**
         * 菜单名称
         **/
        private String menuName;

        /**
         * 菜单类型
         **/
        private String menuType;

        /**
         * 菜单编码
         **/
        private String menuCode;

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

        /**
         * 菜单父级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", "权限管理", "C", "010101", "http://www.baidu.com", "CODE_00", 0));
        array.add(new Tree.Node("CODE_03", "内部管理", "C", "010201", "http://www.baidu.com", "CODE_01", 0));

        array.add(new Tree.Node("CODE_04", "内部用户权限", "C", "010201", "http://www.baidu.com", "CODE_02", 0));
        array.add(new Tree.Node("CODE_05", "外部用户权限", "C", "010201", "http://www.baidu.com", "CODE_02", 0));
        array.add(new Tree.Node("CODE_06", "子权限", "C", "010201", "http://www.baidu.com", "CODE_04", 0));
        array.add(new Tree.Node("CODE_07", "子系统内部管理", "C", "010201", "http://www.baidu.com", "CODE_03", 0));

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

        // 取出当下节点下所有数据
        Node node = new Tree.Node("CODE_01x", "系统管理员", "M", "0101", "#", "0", 0);
        Tree.getNodeJson(node, array);
        System.out.println(JSON.toJSONString(node));
    }
}

解析结果:

[
    {
        "children": [
            {
                "children": [
                    {
                        "children": [],
                        "menuCode": "010201",
                        "menuId": "CODE_07",
                        "menuName": "子系统内部管理",
                        "menuType": "C",
                        "parentId": "CODE_03",
                        "sortNum": 0,
                        "url": "http://www.baidu.com"
                    }
                ],
                "menuCode": "010201",
                "menuId": "CODE_03",
                "menuName": "内部管理",
                "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": "子权限",
                                "menuType": "C",
                                "parentId": "CODE_04",
                                "sortNum": 0,
                                "url": "http://www.baidu.com"
                            }
                        ],
                        "menuCode": "010201",
                        "menuId": "CODE_04",
                        "menuName": "内部用户权限",
                        "menuType": "C",
                        "parentId": "CODE_02",
                        "sortNum": 0,
                        "url": "http://www.baidu.com"
                    },
                    {
                        "children": [],
                        "menuCode": "010201",
                        "menuId": "CODE_05",
                        "menuName": "外部用户权限",
                        "menuType": "C",
                        "parentId": "CODE_02",
                        "sortNum": 0,
                        "url": "http://www.baidu.com"
                    }
                ],
                "menuCode": "010101",
                "menuId": "CODE_02",
                "menuName": "权限管理",
                "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": "#"
    }
]

支持无限级联树,仅需要保证数据结构中menuId与parentId需要存在级联关系即可。个人简单测试过,小伙伴有需要可以再度验证下。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,569评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,499评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,271评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,087评论 0 209
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,474评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,670评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,911评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,636评论 0 202
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,397评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,607评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,093评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,418评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,074评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,092评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,865评论 0 196
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,726评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,627评论 2 270