后端返回树结构

出参结构

java 复制代码
@Getter
@Setter
public class TreeResponse implements Serializable {
    // 主键
    private Long id;
    // 父级节点
    private Long parentId;
    // 层级
    private Byte layer;
    // 编码
    private String docCode;
    // 名称
    private String docName;
    // 子节点
    private List<TreeResponse> childNode;
}

方案一:生撸

java 复制代码
public Result<List<TreeResponse>> listTreeData(String code) {
    // 查询全量数据
    List<TreeResponse> tempTreeResponse = new ArrayList<>();
    // 树节点转换
    return Result.success(convertTreeStructure(tempTreeResponse));
}

/**
 * 树形转换
 *
 * @param tempTreeResponse 档案信息
 * @return 档案信息-树
 */
private List<TreeResponse> convertTreeStructure(List<TreeResponse> tempTreeResponse) {
    // 获取第一层级
	List<TreeResponse> treeResponse = tempTreeResponse.stream()
			.filter(value -> Objects.isNull(value.getParentId()))
			.collect(Collectors.toList());
    // 获取非第一层级数据,并以父级ID进行分组
	Map<Long, List<TreeResponse>> notFirstLayerData = tempTreeResponse.stream()
			.filter(value -> Objects.nonNull(value.getParentId()))
			.collect(Collectors.groupingBy(TreeResponse::getParentId));
	treeResponse.forEach(data -> setChildData(data, notFirstLayerData));
	return treeResponse;
}

/**
 * 设置子节点
 *
 * @param parentInfo        父节点信息
 * @param notFirstLayerData 非首节点信息
 */
private void setChildData(TreeResponse parentInfo, Map<Long, List<TreeResponse>> notFirstLayerData) {
	List<TreeResponse> childData = notFirstLayerData.get(parentInfo.getId());
	if (CollectionUtils.isNotEmpty(childData)) {
		parentInfo.setChildNode(childData);
		childData.forEach(data -> setChildData(data, notFirstLayerData));
	}
}

方案二:Hutool

java 复制代码
//配置
TreeNodeConfig treeNodeConfig = new TreeNodeConfig();
// 自定义属性名 都要默认值的
treeNodeConfig.setWeightKey("order");
treeNodeConfig.setIdKey("rid");
// 最大递归深度
treeNodeConfig.setDeep(3);

//转换器 (含义:找出父节点为字符串零的所有子节点, 并递归查找对应的子节点, 深度最多为 3)
List<Tree<String>> treeNodes = TreeUtil.<TreeNode, String>build(nodeList, "0", treeNodeConfig,
		(treeNode, tree) -> {
			tree.setId(treeNode.getId());
			tree.setParentId(treeNode.getParentId());
			tree.setWeight(treeNode.getWeight());
			tree.setName(treeNode.getName());
			// 扩展属性 ...
			tree.putExtra("extraField", 666);
			tree.putExtra("other", new Object());
		});
相关推荐
Trouvaille ~21 小时前
【优选算法篇】深入浅出链表算法:交换、重排与合并的终极策略
c++·算法·链表·面试·蓝桥杯·笔试·后端开发
墨白曦煜21 小时前
算法实战笔记:链表的底层逻辑与指针的高阶玩法(二)
笔记·算法·链表
风筝在晴天搁浅2 天前
阿里 LeetCode 876.链表的中间节点
算法·leetcode·链表
玖釉-2 天前
二叉树展开为链表:从先序遍历到原地指针重排
c++·windows·算法·leetcode·链表
sheeta19982 天前
LeetCode 补拙笔记 日期:2026.05.27 题目:61. 旋转链表
笔记·leetcode·链表
菜菜的顾清寒3 天前
力扣HOT100(31)K 个一组翻转链表
算法·leetcode·链表
玖釉-3 天前
从有序链表合并看链表算法的指针设计:LeetCode 21「合并两个有序链表」深度解析
c++·windows·算法·leetcode·链表
菜菜的顾清寒4 天前
力扣HOT100(30)两两交换链表中的节点
算法·leetcode·链表
Dlrb12114 天前
数据结构-单链表与双链表
c语言·数据结构·链表·排序·双链表
菜菜的顾清寒4 天前
力扣HOT10(29) 删除链表的倒数第 N 个结点
算法·leetcode·链表