TreeUtils 树工具类

数据展示:

如图:部门树数据  ,树形的基础数据 id  、 parentId 、label 便可形成

嵌套对象字段如下:{id: 103, parentId: 101, label: "研发部门", weight: 1}

一、工具类 

继承了 hutoo 的工具类 TreeUtil 引入hutool 依赖

@NoArgsConstructor(access = AccessLevel.PRIVATE)
public class TreeBuildUtils extends TreeUtil {

    /**
     * 根据前端定制差异化字段
     */
    public static final TreeNodeConfig DEFAULT_CONFIG = TreeNodeConfig.DEFAULT_CONFIG.setNameKey("label");

    /**
     * <简述>
     * <详细描述>
     * @author syf
     * @date 2024/8/15 11:27
     * @param list 提供的集合
     * @param nodeParser 节点解析器
     * @return java.util.List<cn.hutool.core.lang.tree.Tree < K>>
     */
    public static <T, K> List<Tree<K>> build(List<T> list, NodeParser<T, K> nodeParser) {
        if (CollUtil.isEmpty(list)) {
            return null;
        }
        // 获取父节点
        K k = ReflectUtils.invokeGetter(list.get(0), "parentId");
        return TreeUtil.build(list, k, DEFAULT_CONFIG, nodeParser);
    }

}

 TreeNodeConfig 树形需要的字段:可以通过set字段修改

public class TreeNodeConfig implements Serializable {
    private static final long serialVersionUID = 1L;
    public static TreeNodeConfig DEFAULT_CONFIG = new TreeNodeConfig();
    private String idKey = "id";
    private String parentIdKey = "parentId";
    private String weightKey = "weight";
    private String nameKey = "name";
    private String childrenKey = "children";
    private Integer deep;

。。。。。省里get  set

 TreeUtil  方法展示:树形构建已经操作节点方法

二、准备简单树形数据

@Slf4j
public class TestTreeNode {
   static List<TreeNode<Integer>> createTreeNodes(){
       List<TreeNode<Integer>> list = new ArrayList<>();
       TreeNode<Integer>  root= new TreeNode<>();
       root.setName("总公司");
       root.setId(1);
       root.setParentId(0);

       TreeNode<Integer>  node1= new TreeNode<>();
       node1.setName("分子公司1");node1.setId(2);node1.setParentId(1);


       TreeNode<Integer>  node2= new TreeNode<>();
       node2.setName("财务部");node2.setId(3);node2.setParentId(2);
       TreeNode<Integer>  node3= new TreeNode<>();
       node3.setName("开发部").setId(4).setParentId(2);

       list.add(root);
       list.add(node1);
       list.add(node2);
       list.add(node3);
       return list;
   }

    public static void main(String[] args) {
        buildSingle();
    }

    static void  buildSingle(){
        List<TreeNode<Integer>> list = createTreeNodes();
        Tree<Integer> integerTree = TreeBuildUtils.buildSingle(list);
        Console.log(integerTree);
    }


}

 打印结果:

null[0]
  总公司[1]
    分子公司1[2]
      财务部[3]
      开发部[4]

三、测试所有常用树形方法,并附上DEMO演示

  1. 构建树形
  2. 获取指定节点
  3. 添加节点
  4. 移除节点
  5. 遍历
  6. 过滤
  7. 克隆
public static void main(String[] args) {
        buildSingle();
    }

    static void  buildSingle(){
        List<TreeNode<Integer>> list = createTreeNodes();

        //build 对比上面 去掉最外层 null 的父节点
        List<Tree<Integer>> build = TreeBuildUtils.build(list);
        Console.log(build);


        Console.log("-------------------------------------------测试开始------------------------------------------------");

        //原生 构建树结构
        Tree<Integer> integerTree = TreeBuildUtils.buildSingle(list);
        Console.log(integerTree);

        // 获取父节点
        Tree<Integer> parent = integerTree.getParent();
        Console.log("getParent->{}", parent);
        // 获取节点
        Tree<Integer> node = integerTree.getNode(2);
        Console.log("getNode->{}", node);
        // 获取父节点名称
        List<CharSequence> parentsName = integerTree.getParentsName(4, true);
        Console.log("getParentsName->{}", parentsName);
        // 获取配置
        TreeNodeConfig config = integerTree.getConfig();


        // 添加子节点
//        Tree<Integer> integerTree1 = integerTree.addChildren(node);
//        Console.log("addChildren->{}", integerTree1);
        // 判断是否有子节点
        boolean b = integerTree.hasChild();
        Console.log("hasChild->{}", b);
        // 获取子节点
        List<Tree<Integer>> children = integerTree.getChildren();
        Console.log("getChildren->{}", children);

        //遍历整数树结构,并打印每个节点
        integerTree.walk(new Consumer<Tree<Integer>>() {
            @Override
            public void accept(Tree<Integer> integerTree) {
                // 打印节点
                Console.log("walk->{}", integerTree);
            }
        });


        //根据是否包含特定名称(如“分子公司”)来筛选树结构数据
        Tree<Integer> filterNew = integerTree.filterNew(new Filter<Tree<Integer>>() {
            @Override
            public boolean accept(Tree<Integer> integerTree) {
                if (null != integerTree.getName() && integerTree.getName().toString().contains("分子公司")) {
                    return true;
                }
                return false;
            }
        });
        Console.log("filterNew->{}", filterNew);


        // 创建一个过滤器,用于筛选出名称中包含"分子公司"的Tree对象
        Tree<Integer> filter = filterNew.filter(new Filter<Tree<Integer>>() {
            @Override
            public boolean accept(Tree<Integer> integerTree) {
                if (null != integerTree.getName() && integerTree.getName().toString().contains("分子公司")) {
                    return true;
                }
                return false;
            }
        });
        Console.log("filter->{}", filter);


        // 克隆一个整数树结构
        Tree<Integer> integerTree2 = integerTree.cloneTree();
        Console.log("cloneTree->{}", integerTree2);


    }

测试结果:

null[0]
  总公司[1]
    分子公司1[2]
      财务部[3]
      开发部[4]

[总公司[1]
  分子公司1[2]
    财务部[3]
    开发部[4]
]
getParent->null
getNode->分子公司1[2]
  财务部[3]
  开发部[4]

getParentsName->[开发部, 分子公司1, 总公司]
hasChild->true
getChildren->[总公司[1]
  分子公司1[2]
    财务部[3]
    开发部[4]
]
walk->null[0]
  总公司[1]
    分子公司1[2]
      财务部[3]
      开发部[4]

walk->总公司[1]
  分子公司1[2]
    财务部[3]
    开发部[4]

walk->分子公司1[2]
  财务部[3]
  开发部[4]

walk->财务部[3]

walk->开发部[4]

filterNew->null[0]
  总公司[1]
    分子公司1[2]
      财务部[3]
      开发部[4]

filter->null[0]
  总公司[1]
    分子公司1[2]
      财务部[3]
      开发部[4]

cloneTree->null[0]
  总公司[1]
    分子公司1[2]
      财务部[3]
      开发部[4]


Process finished with exit code 0

  • 8
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
当然可以,以下是一个简单的 Java 通用结构转换工具类: ```java import java.util.ArrayList; import java.util.List; public class TreeUtils { /** * 将通用结构转换为指定类型的结构 * * @param sourceList 原始结构列表 * @param rootId 根节点 ID * @param builder 转换器 * @param <T> 目标结构节点类型 * @return 目标结构列表 */ public static <T extends TreeNode> List<T> convertTree(List<? extends TreeNode> sourceList, String rootId, TreeBuilder<T> builder) { List<T> targetList = new ArrayList<>(); for (TreeNode sourceNode : sourceList) { if (rootId.equals(sourceNode.getParentId())) { T targetNode = builder.build(sourceNode); targetNode.setChildren(getChildren(sourceNode.getId(), sourceList, builder)); targetList.add(targetNode); } } return targetList; } /** * 获取指定节点的子节点列表 * * @param parentId 父节点 ID * @param sourceList 原始结构列表 * @param builder 转换器 * @param <T> 目标结构节点类型 * @return 子节点列表 */ private static <T extends TreeNode> List<T> getChildren(String parentId, List<? extends TreeNode> sourceList, TreeBuilder<T> builder) { List<T> children = new ArrayList<>(); for (TreeNode sourceNode : sourceList) { if (parentId.equals(sourceNode.getParentId())) { T targetNode = builder.build(sourceNode); targetNode.setChildren(getChildren(sourceNode.getId(), sourceList, builder)); children.add(targetNode); } } return children; } /** * 结构节点接口 */ public interface TreeNode { /** * 获取节点 ID * * @return 节点 ID */ String getId(); /** * 获取父节点 ID * * @return 父节点 ID */ String getParentId(); } /** * 结构节点构建器接口 * * @param <T> 目标结构节点类型 */ public interface TreeBuilder<T extends TreeNode> { /** * 构建目标结构节点 * * @param sourceNode 原始结构节点 * @return 目标结构节点 */ T build(TreeNode sourceNode); } } ``` 使用方法示例: 假设有一个 `Node` 类,表示通用结构节点,包含 `id`、`parentId`、`name` 等属性。现在需要将 `Node` 类型的通用结构转换为 `Menu` 类型的结构,其中 `Menu` 类型的节点包含 `id`、`parentId`、`name`、`url` 等属性。可以按以下步骤进行转换: 1. 定义 `Menu` 类型的节点: ```java public class Menu implements TreeUtils.TreeNode { private String id; private String parentId; private String name; private String url; private List<Menu> children; // 省略 getter 和 setter 方法 } ``` 2. 定义 `Menu` 类型的节点构建器: ```java public class MenuBuilder implements TreeUtils.TreeBuilder<Menu> { @Override public Menu build(TreeUtils.TreeNode sourceNode) { Node node = (Node) sourceNode; Menu menu = new Menu(); menu.setId(node.getId()); menu.setParentId(node.getParentId()); menu.setName(node.getName()); menu.setUrl(node.getUrl()); return menu; } } ``` 3. 调用 `TreeUtils.convertTree()` 方法进行转换: ```java List<Node> nodeList = ... // 原始结构列表 List<Menu> menuList = TreeUtils.convertTree(nodeList, "root", new MenuBuilder()); ``` 其中,`nodeList` 是原始结构列表,`"root"` 是根节点 ID,`new MenuBuilder()` 是 `Menu` 类型的节点构建器。 希望这个工具类能对你有所帮助!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

syfjava

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值