构建无限层级的树形结构

/ JAVA / 1100浏览

构建无限层级的树形结构

开发过程中,常常需要用到树形结构的场景,创建树形结构可以由服务端产生,也可以由客户端产生,今天我们就来说一下由服务端创建树形结构数据。

数据结构

在数据库表中只需要用到3个字段就可以,那就是id,parentId,name;

工具

话不多说,直接上代码。


import lombok.Data;

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

/**
 * 构建树节点
 *
 * @author Create by yachtcay
 * @date 2019/6/4 16:37
 */
@Data
public class TreeNode {
    /**
     * id
     */
    protected Integer id;
    /**
     * 名称
     */
    protected String name;
    /**
     * 父级id
     */
    protected Integer parentId;
    /**
     * 父级名称
     */
    protected String parentName;
    /**
     * 子集
     */
    protected List<TreeNode> children = new ArrayList<TreeNode>();

    public void add(TreeNode node) {
        children.add(node);
    }
}

import lombok.experimental.UtilityClass;

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

/**
 * 构建树工具类
 *
 * @author Create by yachtcay
 * @date 2019/6/4 16:37
 */
@UtilityClass
public class TreeUtil {
    /**
     * 两层循环实现建树
     *
     * @param treeNodes 传入的树节点列表
     * @return
     */
    public <T extends TreeNode> List<T> buildByLoop(List<T> treeNodes, Object root) {

        List<T> trees = new ArrayList<>();

        for (T treeNode : treeNodes) {

            if (root.equals(treeNode.getParentId())) {
                trees.add(treeNode);
            }

            for (T it : treeNodes) {
                if (it.getParentId() == treeNode.getId()) {
                    if (treeNode.getChildren() == null) {
                        treeNode.setChildren(new ArrayList<>());
                    }
                    it.setParentName(treeNode.getName());
                    treeNode.add(it);
                }
            }
        }
        return trees;
    }

    /**
     * 使用递归方法建树
     *
     * @param treeNodes
     * @return
     */
    public <T extends TreeNode> List<T> buildByRecursive(List<T> treeNodes, Object root) {
        List<T> trees = new ArrayList<T>();
        for (T treeNode : treeNodes) {
            if (root.equals(treeNode.getParentId())) {
                trees.add(findChildren(treeNode, treeNodes));
            }
        }
        return trees;
    }

    /**
     * 递归查找子节点
     *
     * @param treeNodes
     * @return
     */
    public <T extends TreeNode> T findChildren(T treeNode, List<T> treeNodes) {
        for (T it : treeNodes) {
            if (treeNode.getId() == it.getParentId()) {
                if (treeNode.getChildren() == null) {
                    treeNode.setChildren(new ArrayList<>());
                }
                treeNode.add(findChildren(it, treeNodes));
            }
        }
        return treeNode;
    }
}

只需要用到这两个类就可以实现构建无限层级的树形结构。

举个栗子

构建部门的树形结构。

/**
     * 获取部门树 service 实现方法
     *
     * @return
     */
    @Override
    public List<DeptTree> listDeptTrees() {
        return getDeptTree(this.list(Wrappers.<SysDept>query().lambda().orderByAsc(SysDept::getSort)));
    }

    /**
     * 构建部门树
     *
     * @param depts 部门list
     * @return
     */
    private List<DeptTree> getDeptTree(List<SysDept> depts) {
        List<DeptTree> treeList = depts.stream()
                .filter(dept -> !dept.getId().equals(dept.getParentId()))
                .map(dept -> {
                    DeptTree node = new DeptTree();
                    node.setId(dept.getId());
                    node.setParentId(dept.getParentId());
                    node.setName(dept.getName());
                    node.setSort(dept.getSort());
                    return node;
                }).collect(Collectors.toList());
        return TreeUtil.buildByLoop(treeList, CommonConstants.TOP_NODE);
    }