springboot树形布局 支持含糊查询,返回匹配节点和父节点,其他节点不返回 ...

打印 上一主题 下一主题

主题 927|帖子 927|积分 2796

  1. package com.me.meterdemo.ds;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. public class TreeNode {
  5.     private Long id;
  6.     private String name;
  7.     private Long parentId;
  8.     private List<TreeNode> children = new ArrayList<>();
  9.     // 构造方法
  10.     public TreeNode(Long id, String name, Long parentId) {
  11.         this.id = id;
  12.         this.name = name;
  13.         this.parentId = parentId;
  14.     }
  15.     // Getters and Setters
  16.     public Long getId() { return id; }
  17.     public String getName() { return name; }
  18.     public Long getParentId() { return parentId; }
  19.     public List<TreeNode> getChildren() { return children; }
  20.     public void addChild(TreeNode child) {
  21.         // 防止重复添加
  22.         if (children.stream().noneMatch(c -> c.id.equals(child.id))) {
  23.             children.add(child);
  24.         }
  25.     }
  26. }
复制代码

  1. package com.me.meterdemo.ds;
  2. import org.springframework.stereotype.Service;
  3. import javax.annotation.PostConstruct;
  4. import java.util.*;
  5. import java.util.stream.Collectors;
  6. @Service
  7. public class TreeMemoryService {
  8.     // 内存数据存储
  9.     private final Map<Long, TreeNode> nodeMap = new HashMap<>();
  10.     // 初始化测试数据
  11.     @PostConstruct
  12.     private void initData() {
  13.         clearAndInitData();
  14.     }
  15.     private void clearAndInitData() {
  16.         nodeMap.clear();
  17.         addNode(1L, "集团总部", null);
  18.         addNode(2L, "华东公司", 1L);
  19.         addNode(3L, "技术部", 2L);
  20.         addNode(4L, "开发组", 3L);
  21.         addNode(5L, "测试组", 3L);
  22.         addNode(6L, "华南公司", 1L);
  23.     }
  24.     private void addNode(Long id, String name, Long parentId) {
  25.         TreeNode node = new TreeNode(id, name, parentId);
  26.         nodeMap.put(id, node);
  27.     }
  28.     /**
  29.      * 执行树结构搜索
  30.      * @param keyword 查询关键词(可为空)
  31.      * @return 符合条件节点及其父节点组成的树结构
  32.      */
  33.     public List<TreeNode> search(String keyword) {
  34.         // 空关键词返回空结果
  35.         if (keyword != null && keyword.trim().isEmpty()) {
  36.             return Collections.emptyList();
  37.         }
  38.         // 获取所有匹配节点
  39.         List<TreeNode> matches = nodeMap.values().stream()
  40.                 .filter(node -> matchesKeyword(node, keyword))
  41.                 .collect(Collectors.toList());
  42.         // 收集需要展示的节点ID(匹配节点+父链)
  43.         Set<Long> visibleIds = new LinkedHashSet<>();
  44.         matches.forEach(node -> collectParentChain(node, visibleIds));
  45.         // 构建精简树结构
  46.         return buildPrunedTree(visibleIds);
  47.     }
  48.     // 判断节点是否匹配关键词
  49.     private boolean matchesKeyword(TreeNode node, String keyword) {
  50.         return keyword == null ||
  51.                 node.getName().contains(keyword);
  52.     }
  53.     // 收集节点及其所有父节点
  54.     private void collectParentChain(TreeNode node, Set<Long> ids) {
  55.         Long currentId = node.getId();
  56.         while (currentId != null) {
  57.             if (!ids.add(currentId)) break; // 防止循环
  58.             TreeNode current = nodeMap.get(currentId);
  59.             currentId = current.getParentId();
  60.         }
  61.     }
  62.     // 构建修剪后的树结构
  63.     private List<TreeNode> buildPrunedTree(Set<Long> visibleIds) {
  64.         // 创建节点副本防止污染原始数据
  65.         Map<Long, TreeNode> visibleNodes = createNodeCopies(visibleIds);
  66.         // 构建层级关系
  67.         List<TreeNode> roots = new ArrayList<>();
  68.         visibleNodes.values().forEach(node -> {
  69.             Long parentId = node.getParentId();
  70.             if (parentId == null) {
  71.                 roots.add(node);
  72.             } else {
  73.                 TreeNode parent = visibleNodes.get(parentId);
  74.                 if (parent != null) {
  75.                     parent.addChild(node);
  76.                 }
  77.             }
  78.         });
  79.         return roots;
  80.     }
  81.     // 创建节点副本(清空原有子节点)
  82.     private Map<Long, TreeNode> createNodeCopies(Set<Long> visibleIds) {
  83.         return visibleIds.stream()
  84.                 .map(id -> nodeMap.get(id))
  85.                 .filter(Objects::nonNull)
  86.                 .collect(Collectors.toMap(
  87.                         TreeNode::getId,
  88.                         orig -> new TreeNode(orig.getId(),
  89.                                 orig.getName(),
  90.                                 orig.getParentId())
  91.                 ));
  92.     }
  93.     // 重置测试数据(可选)
  94.     public void resetData() {
  95.         clearAndInitData();
  96.     }
  97. }
复制代码

  1. package com.me.meterdemo.ds;
  2. import org.springframework.web.bind.annotation.GetMapping;
  3. import org.springframework.web.bind.annotation.RequestParam;
  4. import org.springframework.web.bind.annotation.RestController;
  5. import java.util.List;
  6. @RestController
  7. public class TreeDSController {
  8.     private final TreeMemoryService treeMemoryService;
  9.     public TreeDSController(TreeMemoryService treeService) {
  10.         this.treeMemoryService = treeService;
  11.     }
  12.     @GetMapping("/search")
  13.     public List<TreeNode> searchTree(@RequestParam(required = false) String keyword) {
  14.         return treeMemoryService.search(keyword);
  15.     }
  16. }
复制代码
springboot树形布局 支持含糊查询,返回匹配节点和父节点,其他节点不返回

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

大连全瓷种植牙齿制作中心

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表