01-线上问题处理-树形结构拼接

1 递归方式实现

1.1 代码

步骤一:一次性获取到所有的数据tisManualCatalogPOS

步骤二:获取到一级节点(parentId为null)

步骤三:递归向下拼接
树的深度很深,导致堆栈越多,可能

生产8K的数据能很明显的看到接口响应慢,<没打印日志>

复制代码
  private List<ManualTreeVO> buildTree(List<TisManualCatalogPO> tisManualCatalogPOS) {
        List<ManualTreeVO> treeVOList = Lists.newArrayList();
        if (CollectionUtils.isEmpty(tisManualCatalogPOS)) {
            return treeVOList;
        }

        List<ManualTreeVO> dataList = tisManualCatalogPOS.stream().map(x -> {
            ManualTreeVO bean = new ManualTreeVO();
            bean.setId(x.getId());
            bean.setName(x.getName());
            bean.setSort(x.getSort());
            bean.setType(x.getType());
            bean.setCode(x.getCode());
            bean.setParentId(x.getParentId());
            String filePath = x.getFilePath();
            // 不包含则拼前缀
            if (StringUtils.isNotEmpty(filePath) && !filePath.contains(filePathPrefix)) {
                filePath = filePathPrefix + filePath;
            }
            bean.setFilePath(filePath);
            return bean;
        }).collect(Collectors.toList());

        // 拿到所有的一级节点
        List<ManualTreeVO> firstCatalogList = dataList.stream()
                .filter(x -> Objects.isNull(x.getParentId()))
                .sorted(Comparator.comparing(ManualTreeVO::getSort))
                .collect(Collectors.toList());

        for (ManualTreeVO data : firstCatalogList) {
            // 构建树
            doBuildTree(dataList, data);
            treeVOList.add(data);
        }

        return treeVOList;
    }

    private void doBuildTree(List<ManualTreeVO> hoursCatalogList, ManualTreeVO treeVO) {
        // 下一级
        List<ManualTreeVO> childrenList = getChildrenList(hoursCatalogList, treeVO);
        treeVO.setChildrenList(childrenList);

        for (ManualTreeVO children : childrenList) {
            // 有下级
            if (!CollectionUtils.isEmpty(getChildrenList(hoursCatalogList, children))) {
                doBuildTree(hoursCatalogList, children);
            }
        }
    }

    private List<ManualTreeVO> getChildrenList(List<ManualTreeVO> hoursCatalogList, ManualTreeVO treeVO) {
        // 下一级
        List<ManualTreeVO> nextList = hoursCatalogList.stream().filter(x -> Objects.equals(treeVO.getCode(), x.getParentId()))
                .sorted(Comparator.comparing(ManualTreeVO::getSort))
                .collect(Collectors.toList());
        return nextList;
    }

1.2 性能

2 非递归方式实现(推荐)

2.1 代码

使用Map,哈希的方式用空间换时间,一次遍历即可拼接成树

复制代码
    public List<ManualTreeVO> buildTreeNew(String uuid, List<TisManualCatalogPO> tisManualCatalogPOS) {
        log.info("手册目录-拼接目录 uuid {} {}", uuid, CollectionUtils.isNotEmpty(tisManualCatalogPOS) ? tisManualCatalogPOS.size() : 0);
        List<ManualTreeVO> roots = Lists.newArrayList();
        Map<String, ManualTreeVO> nodeMap = new HashMap<>();
        List<ManualTreeVO> dataList = tisManualCatalogPOS.stream().map(x -> {
            ManualTreeVO bean = new ManualTreeVO();
            bean.setId(x.getId());
            bean.setName(x.getName());
            bean.setSort(x.getSort());
            bean.setType(x.getType());
            bean.setCode(x.getCode());
            bean.setParentId(x.getParentId());
            String filePath = x.getFilePath();
            // 不包含则拼前缀
            if (StringUtils.isNotEmpty(filePath) && !filePath.contains(filePathPrefix)) {
                filePath = filePathPrefix + filePath;
            }
            bean.setFilePath(filePath);
            return bean;
        }).collect(Collectors.toList());
        // 将所有节点存入映射表
        for (ManualTreeVO node : dataList) {
            nodeMap.put(node.getCode(), node);
        }

        // 构建树结构
        for (ManualTreeVO node : dataList) {
            String parentId = node.getParentId();
            if (parentId == null || parentId.isEmpty()) {
                roots.add(node);
            } else {
                ManualTreeVO parent = nodeMap.get(parentId);
                if (parent != null) {
                    parent.addChild(node);
                } else {
                    // 如果父节点不存在,将其作为根节点
                    roots.add(node);
                }
            }
        }
        log.info("手册目录-拼接目录 uuid end {}", uuid);
        return roots;
    }

2.2 性能

stg日志:3662条数据拼接成树花费时间

3 优劣对比

使用 Map 非递归实现相比递归实现有以下优势:

  1. 避免栈溢出:递归实现在处理深度很大的树时可能导致栈溢出,而非递归实现没有这个问题

  2. 性能更好:非递归实现通常比递归实现有更好的性能,特别是对于大数据集

  3. 更易调试:非递归实现的执行流程更线性,更容易调试和理解

  4. 内存控制:非递归实现可以更好地控制内存使用,特别是在使用迭代器或队列时

  5. 可中断性:非递归实现可以更容易地添加中断条件或超时机制

4 使用建议

  1. 小规模数据:可以使用简单的递归实现,代码更简洁

  2. 大规模数据:推荐使用非递归实现,避免栈溢出风险

  3. 性能关键:使用高效的单次遍历实现,性能最佳

  4. 内存敏感:使用基于栈的实现,内存使用更可控

相关推荐
Hello.Reader6 小时前
PyFlink JAR、Python 包、requirements、虚拟环境、模型文件,远程集群怎么一次搞定?
java·python·jar
计算机学姐6 小时前
基于SpringBoot的汽车租赁系统【个性化推荐算法+数据可视化统计】
java·vue.js·spring boot·后端·spring·汽车·推荐算法
七夜zippoe6 小时前
分布式事务解决方案 2PC 3PC与JTA深度解析
java·分布式事务·cap·2pc·3pc·jta
我是人✓6 小时前
Spring IOC入门
java·数据库·spring
好好研究6 小时前
SpringBoot小案例打包执行流程
java·spring boot·后端
rgeshfgreh6 小时前
Spring Bean管理机制深度解析
java·spring boot·spring
ling-456 小时前
ssm-day07 springboot3、Mybatis-Plus、springboot实战
java·spring boot·后端
少许极端6 小时前
算法奇妙屋(二十三)-完全背包问题(动态规划)
java·算法·动态规划·完全背包
a程序小傲6 小时前
得物Java面试被问:边缘计算的数据同步和计算卸载
java·开发语言·数据库·后端·面试·golang·边缘计算
你不是我我6 小时前
【Java 开发日记】我们来说一下无锁队列 Disruptor 的原理
java·开发语言