Java实现构建树状菜单栏

在这里插入图片描述

一. 问题背景

在公司做项目遇到一个需求是要返回一颗树状菜单栏的数据结构给前端。效果类似如下:

在这里插入图片描述

二. 解决方案

这里先直接给出核心算法的代码,有兴趣研究的可以前往gitee页面下载,搭建该项目的所有资源(包括源码、sql表结构、表数据都在项目里面)都在里面,部署方法可参考里面的readme.md文件

2.1 核心算法

这里有2种方式,其实都差不多。一种是采用 动态数组 ,一种是采用 队列层次遍历 。直接给出核心代码,后面再给出思路图

2.1.1 用动态数组构建树

 		/**
         * 1. 构建一个map,存储每个节点的孩子们
         * 2. 构建一个map,根据key可以获取到这个节点
         * 3. 根据传进来的字典定义名获取那个节点,用动态数组构建树
         */

        long startTime = System.currentTimeMillis();

        if (parentKey == null || parentKey.trim().length() == 0) {
            parentKey = PARENT_KEY;
        }

        // 构建一个map,存储每个节点的孩子节点
        Map<Integer, List<DictDefine>> parentIdChildrenMap = new HashMap<>();
        // 构建一个map,根据key可以获取到这个节点
        Map<String, DictDefine> defineNameDictDefineMap = new HashMap<>();
        for (DictDefine dictDefine : dictDefines) {
            Integer parentId = dictDefine.getParentId();
            List<DictDefine> dictDefineList = parentIdChildrenMap.get(parentId);
            if (dictDefineList == null) {
                dictDefineList = new ArrayList<>();
            }
            dictDefineList.add(dictDefine);
            parentIdChildrenMap.put(parentId, dictDefineList);
            defineNameDictDefineMap.put(dictDefine.getDefineName(), dictDefine);
        }

        // 根据传进来的字典定义名获取那个节点,用动态数组构建树
        DictDefine rootDictDefine = defineNameDictDefineMap.get(parentKey);

        if (rootDictDefine == null) {
            return new ArrayList<DictDefine>();
        }

        //用动态数组list构建树
        List<DictDefine> list = new ArrayList<>();
        list.add(rootDictDefine);

        int index = 0;
        while (!list.isEmpty() && index < list.size()) {
            DictDefine dictDefine = list.get(index);
            if (dictDefine == null) {
                continue;
            }
            List<DictDefine> childrenList = parentIdChildrenMap.get(dictDefine.getId());
            if (childrenList != null) {
                dictDefine.setChildren(childrenList);
                list.addAll(childrenList);
            }

            index++;
        }

        ArrayList<DictDefine> result = new ArrayList<>();
        result.add(rootDictDefine);

        long endTime = System.currentTimeMillis();

        log.info("It takes {}ms to deal it.", endTime-startTime);

        return result;
    }

2.1.2 用队列构建树

 /**
     * 使用队列构建树
     * @param dictDefines
     * @param parentKey
     * @return
     */
    public static List<DictDefine> buildTreeByLinkedList(List<DictDefine> dictDefines,
                                                        String parentKey) {

        /**
         * 1. 构建一个map,存储每个节点的孩子们
         * 2. 构建一个map,根据key可以获取到这个节点
         * 3. 根据传进来的字典定义名获取那个节点,用队列构建树
         */

        long startTime = System.currentTimeMillis();

        if (parentKey == null || parentKey.trim().length() == 0) {
            parentKey = PARENT_KEY;
        }

        // 构建一个map,存储每个节点的孩子节点
        Map<Integer, List<DictDefine>> parentIdChildrenMap = new HashMap<>();
        // 构建一个map,根据key可以获取到这个节点
        Map<String, DictDefine> defineNameDictDefineMap = new HashMap<>();
        for (DictDefine dictDefine : dictDefines) {
            Integer parentId = dictDefine.getParentId();
            List<DictDefine> dictDefineList = parentIdChildrenMap.get(parentId);
            if (dictDefineList == null) {
                dictDefineList = new ArrayList<>();
            }
            dictDefineList.add(dictDefine);
            parentIdChildrenMap.put(parentId, dictDefineList);
            defineNameDictDefineMap.put(dictDefine.getDefineName(), dictDefine);
        }

        // 根据传进来的字典定义名获取那个节点,用动态数组构建树
        DictDefine rootDictDefine = defineNameDictDefineMap.get(parentKey);

        if (rootDictDefine == null) {
            return new ArrayList<DictDefine>();
        }

        //用动态数组list构建树
        Queue<DictDefine> queue = new LinkedList<>();
        queue.offer(rootDictDefine);

        while (!queue.isEmpty() ) {
            DictDefine dictDefine = queue.poll();
            if (dictDefine == null) {
                continue;
            }
            List<DictDefine> childrenList = parentIdChildrenMap.get(dictDefine.getId());
            if (childrenList != null) {
                dictDefine.setChildren(childrenList);
                queue.addAll(childrenList);
            }

        }

        ArrayList<DictDefine> result = new ArrayList<>();
        result.add(rootDictDefine);

        long endTime = System.currentTimeMillis();

        log.info("It takes {}ms to deal it.", endTime-startTime);

        return result;
    }

2.2 核心思路

核心思路如上,不管是采用动态数组,还是队列,其实都是在进行层次遍历,有兴趣可以去看看BFS层次遍历算法

在这里插入图片描述

  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Java漂亮的形菜单制作实例(源码),森林状的关系图,文本域,用于显示点击的节点名称,使用了JTree,可以看作是一个jTree的用法演示实例。形菜单应用广泛,这个Tree制作漂亮,相信会让很多朋友从中学习到实现的方法,效果如演示截图所示。下面是相关的代码片段:   DefaultMutableTreeNode root = new DefaultMutableTreeNode("设置"); //生成根节点   DefaultMutableTreeNode node1=new DefaultMutableTreeNode("常规"); //生成节点一   node1.add(new DefaultMutableTreeNode("默认路径")); //增加新节点到节点一上   node1.add(new DefaultMutableTreeNode("保存选项"));   root.add(node1); //增加节点一到根节点上   root.add(new DefaultMutableTreeNode("界面"));   root.add(new DefaultMutableTreeNode("提示声音"));   root.add(new DefaultMutableTreeNode("打印"));   JTree tree = new JTree(root); //得到JTree的实例   DefaultTreeCellRenderer renderer = (DefaultTreeCellRenderer)tree.getCellRenderer(); //得到JTree的Renderer   renderer.setLeafIcon(null); //设置叶子节点图标为空   renderer.setClosedIcon(null); //设置关闭节点的图标为空   renderer.setOpenIcon(null); //设置打开节点的图标为空   tree.addTreeSelectionListener(new TreeSelectionListener() {//选择节点的事件处理   public void valueChanged(TreeSelectionEvent evt) {   TreePath path = evt.getPath(); //得到选择路径   String info=path.getLastPathComponent().toString(); //得到选择的节点名称   jtfInfo.setText(info); //在文本域中显示名称   }
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值