递归思想解决树形菜单问题

其实树形菜单的问题我们其实很常见,平常权限管理中的菜单就是最好应用,及时记录下来,后面碰到类似的问题也能及时更好地解决!
在程序中,所谓的递归,就是函数自己直接或间接的调用自己。调用自己分两种:

  • 直接调用自己
  • 间接调用自己

就递归而言最重要的就是跳出结构,因为跳出了才可以有结果.所有要有一个条件判断!
比如:本例中就是判断当前菜单的下一级菜单集合为空

所以我们的树形菜单看做一个类似树形的集合,拆分出来就是分析一棵树
最后菜单的树形递归其实可以看做是二叉树的前序遍历:
按照根节点->左子树->右子树的顺序访问二叉树

image.png

先序遍历:(1)访问根节点;(2)采用先序递归遍历左子树;(3)采用先序递归遍历右子树;
(注:每个节点的分支都遵循上述的访问顺序,体现“递归调用”)
先序遍历结果:A BDFE CGHI

思维过程:
(1)先访问根节点A,
(2)A分为左右两个子树,因为是递归调用,所以左子树也遵循“先根节点-再左-再右”的顺序,所以访问B节点,
(3)然后访问D节点,
(4)访问F节点的时候有分支,同样遵循“先根节点-再左--再右”的顺序,
(5)访问E节点,此时左边的大的子树已经访问完毕,
(6)然后遵循最后访问右子树的顺序,访问右边大的子树,右边大子树同样先访问根节点C,
(7)访问左子树G,
(8)因为G的左子树没有,所以接下俩访问G的右子树H,
(9)最后访问C的右子树I

下面直接贴上代码,为了方便没有采用数据库,主要是要学会递归的思想:

Menu菜单类实体类:

package com.apple;

import java.util.List;

/**
 * @description: 菜单类
 * @author: Apple
 * @create: 2019-07-13 23:13
 **/
public class Menu {
    private int        id;
    private String     name;
    private int        parentId;
    private List<Menu> childMenus;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getParentId() {
        return parentId;
    }

    public void setParentId(int parentId) {
        this.parentId = parentId;
    }

    public List<Menu> getChildMenus() {
        return childMenus;
    }

    public void setChildMenus(List<Menu> childMenus) {
        this.childMenus = childMenus;
    }

    @Override
    public String toString() {
        return "Menu{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", parentId=" + parentId +
                ", childMenus=" + childMenus +
                '}';
    }
}

MenuDao实体操作类,包含了对菜单的一些操作:

package com.apple.rmi;

import java.util.ArrayList;
import java.util.List;

/**
 * @description: 菜单操作类
 * @author: Apple
 * @create: 2019-07-13 23:16
 **/
public class MenuDao {
    /**
     * @return 返回所有的菜单集合
     */
    public List<Menu> getAllMenus() {
        ArrayList<Menu> allMenus = new ArrayList();
        //构建三个一级菜单
        Menu menu = new Menu();
        menu.setId(1);
        menu.setName("一级菜单-1");
        menu.setParentId(0);
        Menu menu2 = new Menu();
        menu2.setId(2);
        menu2.setName("一级菜单-2");
        menu2.setParentId(0);
        Menu menu3 = new Menu();
        menu3.setId(3);
        menu3.setName("一级菜单-3");
        menu3.setParentId(0);

        Menu menu4 = new Menu();
        menu4.setId(4);
        menu4.setName("二级菜单-1");
        menu4.setParentId(1);  //设置在一级菜单-1 下面
        Menu menu5 = new Menu();
        menu5.setId(5);
        menu5.setName("二级菜单-2");
        menu5.setParentId(1);  //设置在一级菜单-1 下面

        Menu menu6 = new Menu();
        menu6.setId(6);
        menu6.setName("三级菜单-1");
        menu6.setParentId(4);  //设置在二级菜单-1 下面
        Menu menu7 = new Menu();
        menu7.setId(7);
        menu7.setName("二级菜单-3");
        menu7.setParentId(2);  //设置在一级菜单-2 下面
        //添加所有菜单
        allMenus.add(menu);
        allMenus.add(menu2);
        allMenus.add(menu3);
        allMenus.add(menu4);
        allMenus.add(menu5);
        allMenus.add(menu6);
        allMenus.add(menu7);
        return allMenus;
    }

    /**
     * @return 返回一级菜单集合
     */
    public List<Menu> getOneMenuList() {
        List<Menu> allMenus    = getAllMenus();
        List<Menu> oneMenuList = new ArrayList<Menu>();
        for (Menu menu : allMenus) {
            //如果当前菜单的parentId为0代表是顶级(一级)菜单
            if (menu.getParentId() == 0) {
                oneMenuList.add(menu);
            }
        }
        return oneMenuList;
    }

    /**
     * @param id
     * @return 返回下一层菜单集合
     */
    public List<Menu> getNextMenuList(int id) {
        List<Menu> allMenus     = getAllMenus();
        List<Menu> nextMenuList = new ArrayList<Menu>();
        for (Menu menu : allMenus) {
            if (menu.getParentId() == id) {  //判断上一层菜单的id是否是下一层的parentId
                nextMenuList.add(menu);
            }
        }
        return nextMenuList;
    }

    /**
     * 递归调用
     *
     * @param menuList
     */
    public void getMenuTree(List<Menu> menuList) {
        //第一次获得的是一级菜单集合,后面传入的是下一级菜单的集合
        List<Menu> oneMenuList = menuList;
        for (int i = 0; i < oneMenuList.size(); i++) {
            Menu menu = oneMenuList.get(i);
            //获得下一级的菜单集合
            List<Menu> nextMenuList = getNextMenuList(menu.getId());
            //递归终止条件判断,如果当前菜单的下一级菜单集合为空,则当前菜单为最后一级菜单
            if (nextMenuList.size() != 0) {
                //设置当前菜单的下一级菜单列表
                menu.setChildMenus(nextMenuList);
                //递归调用第二层的
                getMenuTree(nextMenuList);
            }     
        }
    }

    public static void main(String[] args) {
        MenuDao menuDao = new MenuDao();
        //先获得顶级菜单集合
        List<Menu> menuList = menuDao.getOneMenuList();
        //递归调用,最终目的是为了获得每一个Menu当中的childMenus
        menuDao.getMenuTree(menuList);
        for (Menu menu : menuList) {
            System.out.println(menu.toString());
        }
    }

}

最后的输出结果如下所示:

Menu{
  id=1,
  name='一级菜单-1',
  parentId=0,
  childMenus=[
    Menu{
      id=4,
      name='二级菜单-1',
      parentId=1,
      childMenus=[
        Menu{
          id=6,
          name='三级菜单-1',
          parentId=4,
          childMenus=null
        }
      ]
    },
    Menu{
      id=5,
      name='二级菜单-2',
      parentId=1,
      childMenus=null
    }
  ]
}
Menu{
  id=2,
  name='一级菜单-2',
  parentId=0,
  childMenus=[
    Menu{
      id=7,
      name='二级菜单-3',
      parentId=2,
      childMenus=null
    }
  ]
}
Menu{
  id=3,
  name='一级菜单-3',
  parentId=0,
  childMenus=null
}

总结:
  其实我们所做的递归就是为了得到Menu实体类中的下一级子菜单childMenus,所以我们在平常分析问题的时候应该从本质出发,先多花一点时间想清楚问题,不要一上来就直接写代码,然后逐步思考用什么方法解决,思路最重要,有了思路转化为代码就是一件很简单的事情了

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

推荐阅读更多精彩内容