JAVA 樹形菜單遞歸實現(xiàn)(省市區(qū)同理)

菜單樹形結(jié)構(gòu)

package com.yun.entity.vo;

import io.swagger.annotations.ApiModelProperty;
import lombok.Data;

import java.io.Serializable;
import java.util.ArrayList;
import java.util.List;

@Data
public class MenuVO implements Serializable {

    private static final long serialVersionUID = 1L;

    @ApiModelProperty(value = "主鍵")
    private Integer id;

    @ApiModelProperty(value = "菜單名稱")
    private String name;

    @ApiModelProperty(value = "菜單圖標(biāo)")
    private String icon;

    @ApiModelProperty(value = "菜單路由")
    private String link;

    @ApiModelProperty(value = "菜單索引")
    private String indexTag;

    @ApiModelProperty(value = "菜單類型斥杜,目錄:12801001窖贤,菜單:12801002,按鈕:12801003")
    private Integer type;

    @ApiModelProperty(value = "排序")
    private Integer sort;

    @ApiModelProperty(value = "狀態(tài)钝诚,啟用:12781001棘劣,停用:12781002")
    private Integer status;

    @ApiModelProperty(value = "權(quán)限標(biāo)識")
    private String permission;

    @ApiModelProperty(value = "父節(jié)點主鍵椰拒,-1為根節(jié)點")
    private Integer parentId;

    @ApiModelProperty(value = "用于前端顯示開關(guān)嫉称,status=12781001為true")
    private boolean enable;

    @ApiModelProperty(value = "菜單子表")
    private List<MenuVO> children = new ArrayList<>();
}

package com.yun.service.impl;

import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.yun.constant.DictConstant;
import com.yun.entity.dto.UserDTO;
import com.yun.entity.po.MenuPO;
import com.yun.entity.vo.MenuVO;
import com.yun.mapper.MenuMapper;
import com.yun.service.MenuService;
import org.springframework.beans.BeanUtils;
import org.springframework.stereotype.Service;
import org.springframework.util.CollectionUtils;

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

/**
 * <p>
 * 菜單表 服務(wù)實現(xiàn)類
 * </p>
 *
 * @author CNSTT
 * @since 2023-01-21
 */
@Service
public class MenuServiceImpl extends ServiceImpl<MenuMapper, MenuPO> implements MenuService {

    /**
     * 菜單樹形查詢
     *
     * @param dto 傳參
     * @return List<MenuVO>
     */
    @Override
    public List<MenuVO> searchDefault(MenuDTO dto) {
        LambdaQueryWrapper<MenuPO> wrapper = new QueryWrapper<MenuPO>().lambda();
        // 查詢菜單或目錄
        wrapper.in(MenuPO::getType, DictConstant.DICT_MENU_TYPE_CATALOG, DictConstant.DICT_MENU_TYPE_MENU);
        wrapper.eq(MenuPO::getDeleteFlag, DictConstant.DICT_NO);
        List<MenuPO> menuList = baseMapper.selectList(wrapper);
        if (!CollectionUtils.isEmpty(menuList)) {
            // 樹形返回
            return packageListFunc(menuList);
        }
        return new ArrayList<>();
    }

    /**
     * 組裝菜單樹形結(jié)構(gòu)主方法
     *
     * @param menuList 全量數(shù)據(jù)
     * @return List<MenuVO>
     */
    private List<MenuVO> packageListFunc(List<MenuPO> menuList) {
        List<MenuVO> result = new ArrayList<>();
        List<MenuVO> list = convertResult(menuList);

        for (MenuVO node : list) {
            // parentId = -1為頂層目錄或菜單
            if (node.getParentId() == -1) {
                result.add(findChildren(node, list));
            }
        }
        return result;
    }

    /**
     * 遞歸查詢 - 根據(jù)父節(jié)點獲取子節(jié)點
     *
     * @param node 父節(jié)點
     * @param result 全量數(shù)據(jù)
     * @return MenuVO 填充完父節(jié)點
     */
    private MenuVO findChildren(MenuVO node, List<MenuVO> result) {
        for (MenuVO dto : result) {
            // 父節(jié)點的ID等于子節(jié)點的父節(jié)點ID
            if (node.getId().equals(dto.getParentId())) {
                if (node.getChildren() == null) {
                    node.setChildren(new ArrayList<>());
                }
                node.getChildren().add(findChildren(dto, result));
            }
        }
        return node;
    }

    /**
     * 公共方法 - 實體轉(zhuǎn)化VO出參集合
     *
     * @param menuList PO結(jié)果集
     * @return List<MenuVO>
     */
    private List<MenuVO> convertResult(List<MenuPO> menuList) {
        List<MenuVO> result = new ArrayList<>();
        if (!CollectionUtils.isEmpty(menuList)) {
            menuList.forEach(menuPO -> {
                MenuVO vo = new MenuVO();
                BeanUtils.copyProperties(menuPO, vo);
                vo.setEnable(vo.getStatus() != null && vo.getStatus() == DictConstant.DICT_YES);
                result.add(vo);
            });
        }
        return result;
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末玄妈,一起剝皮案震驚了整個濱河市乾吻,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌拟蜻,老刑警劉巖绎签,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異酝锅,居然都是意外死亡诡必,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進店門搔扁,熙熙樓的掌柜王于貴愁眉苦臉地迎上來爸舒,“玉大人,你說我怎么就攤上這事稿蹲∨っ悖” “怎么了?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵苛聘,是天一觀的道長涂炎。 經(jīng)常有香客問我,道長设哗,這世上最難降的妖魔是什么唱捣? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮网梢,結(jié)果婚禮上震缭,老公的妹妹穿的比我還像新娘。我一直安慰自己澎粟,他們只是感情好蛀序,可當(dāng)我...
    茶點故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布欢瞪。 她就那樣靜靜地躺著,像睡著了一般徐裸。 火紅的嫁衣襯著肌膚如雪遣鼓。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天重贺,我揣著相機與錄音骑祟,去河邊找鬼。 笑死气笙,一個胖子當(dāng)著我的面吹牛次企,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播潜圃,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼缸棵,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了谭期?” 一聲冷哼從身側(cè)響起堵第,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎隧出,沒想到半個月后踏志,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡胀瞪,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年针余,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片凄诞。...
    茶點故事閱讀 38,161評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡圆雁,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出帆谍,到底是詐尸還是另有隱情摸柄,我是刑警寧澤,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布既忆,位于F島的核電站,受9級特大地震影響嗦玖,放射性物質(zhì)發(fā)生泄漏患雇。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一宇挫、第九天 我趴在偏房一處隱蔽的房頂上張望苛吱。 院中可真熱鬧,春花似錦器瘪、人聲如沸翠储。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽援所。三九已至庐舟,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間住拭,已是汗流浹背挪略。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留滔岳,地道東北人杠娱。 一個月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像谱煤,于是被迫代替她去往敵國和親摊求。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,916評論 2 344

推薦閱讀更多精彩內(nèi)容

  • 筆記 Docker Docker怎么實現(xiàn)容器之間交互 network來創(chuàng)建一個橋接網(wǎng)絡(luò)刘离,在docker run的時...
    Cola_Aone閱讀 297評論 0 0
  • DOM對象 WebAPI:瀏覽器提供的 一套通過JS語法操作 瀏覽器和 頁面元素的方法 BOM 和DOM DOM:...
    瘋狂程序猿001閱讀 141評論 0 1
  • 1. UIViewController 生命周期 initWithCoder:通過 nib 文件初始化時觸發(fā)室叉。 a...
    琦均煞Sylar閱讀 214評論 0 0
  • 涉及內(nèi)容來源:《圖解TCP/IP》第一章 網(wǎng)絡(luò)基礎(chǔ)知識 計算機網(wǎng)絡(luò)出現(xiàn)的背景 單機(獨立)模式 計算機設(shè)計之初是解...
    esrever閱讀 235評論 0 0
  • 1、畫兩個相交的圓寥闪,畫圖疊加太惠? 剛開始沒明白面試官的用意,后來想了下疲憋,應(yīng)該是問兩個相交的圓的部分的樣式吧凿渊。andr...
    路過問好閱讀 267評論 0 0