学弟问我如何用后台实现以树形显示分类的业务逻辑,我直接……

昨天 、一个好久没联系的小学弟找到我,说是最近做项目的时候遇到了一个难题,希望我能帮帮他们。本来想草草的推脱、这……我一听是难题,很快啊,我直接来了兴趣……
在这里插入图片描述
平常最大的喜好就是被难题摁在地上摩擦的我,瞬间这嘴角根本就控制不住的上扬
在这里插入图片描述
小学弟:老哥…控制一下……控制一下…
我:咳咳……你且把这难题说给我听听
小学弟:最近做项目的时候,需求是要把分类的结构以树形的样式显示在页面上
我:树形?!嗯……用一下循环嵌套,把二级分类列表嵌套进与其对应的一级分类里面
小学弟:嗯………
在这里插入图片描述
我:害、看老哥给你操作一波
小学弟:geigie~ 你好勇喔~
在这里插入图片描述
其实以树形的样式显示在页面,这个需求并不是很难,简单来说就是将一对多的逻辑关系再加上循环嵌套的问题,复杂的说,就是将一对多的逻辑关系……
在这里插入图片描述
首先,拿到需求先对其进行逻辑分析,分类以树形 ?那么问题来了,何为树形?前端的树形样式排版、拿到后端,用Java如何实现其中的业务逻辑关系 ?来画个图分析一下:
在这里插入图片描述
根据平时对分类等的业务逻辑进行分析,首先这二者是一对多的关系,在一级分类中添加一个盛装二级分类的集合,可以达到一级分类中可以有多个同属性的二级分类,然后在其基础上在套上一层循环,基本上就可以达成树形的结果。理论结束,接下来用代码实现:

第一步:先创建实体类

public class Category implements Serializable {

    private static final long serialVersionUID = 1L;
	//分类id
    private String id;
	//分类名称
    private String title;
	//父ID  一级分类的父ID为:0 , 二级分类父ID = 一级分类的id值    
    private String parentId;
    .......get、set 方法 省略.......
}

为了分析简单,分别再创建出一级和二级各自的实体类

public class OneCategory {

    private String id;
    private String title;

    //一级分类中有多个二级分类
    private List<TwoCategory> children = new ArrayList<>();

	get set 方法省略........

}
public class TwoCategory{

    private String id;
    private String title;
	get set 方法省略........
}

第二步:编写Controller控制层

    //分类列表(树形)
    @GetMapping("/getAllSubject")
    public R getAllSubject(){
        //List集合中的泛型,因为在一级分类中含有一级分类和二级分类
        List<OneCategory> list = subjectService.getAllOneTwoSubject();
        return R.ok().data("list",list);
    }

第三步:编写具体的业务逻辑(为了方便这里选择使用Mybatis-plus

   @Override
    public List<OneCategory> getAllOneTwoSubject() {

        //查询出所有的一级分类 parent_id = 0
        QueryWrapper<Category> oneWrapper = new QueryWrapper<>();
        oneWrapper.eq("parent_id","0");
        List<Category> oneCategoryList = baseMapper.selectList(oneWrapper);

        //查询出所有的二级分类 parent_id != 0
        QueryWrapper<Category> twoWrapper = new QueryWrapper<>();
        twoWrapper.ne("parent_id","0");
        List<Category> twoCategoryList = baseMapper.selectList(twoWrapper);

        //创建一个封装一级分类的集合
        List<OneCategory> finalCategoryList = new ArrayList<>();
        //封装所有的一级分类
        //将查询出来的所有一级分类进行遍历,得到每一个一级分类的对象
        //封装到要求的List集合中
        for (int i = 0; i < oneCategoryList .size(); i++) {
            //得到遍历出来的每一个一级分类的值
            Category oCategory= oneSubjectList.get(i);
            //最终是要将所有的遍历的一级分类的值填充到OneSubject中
            OneCategory oneCategory = new OneCategory ();
            //oCategory值复制到对应的oneCategory 对象里面
            //对于BeanUtils.copyProperties()下面会具体介绍
            BeanUtils.copyProperties(oCategory,oneCategory);

            finalCategoryList.add(oneCategory);


            //创建一个盛装二级分类的集合
            List<TwoCategory> twoFinalCateList = new ArrayList<>();
            //封装所有的二级分类
            //将所有的二级分类遍历出来,得到的每一个二级分类的值封装到要求的List中
            for (int m = 0; m < twoCategoryList.size(); m++) {
                //遍历获取每一个二级分类
                Category tCategory = twoCategoryList.get(m);
                //判断二级分类的parentId是否与一级分类的id相等
                if (tCategory .getParentId().equals(oneSubject.getId())){
                    //把tSubject中的值复制到twoSubject中去,然后在放到twoFinalSubject中
                    TwoSubject twoCategory = new TwoSubject();
                    BeanUtils.copyProperties(tCategory,twoCategory );
                    twoFinalSubjectList.add(twoSubject);
                }
            }
            //最后再把相应一级分类下所有的二级分类放到一级分类中
            OneCategory.setChildren(twoFinalCategoryList);
        }
        return finalCategoryList;
    }

以上就是如何以树形样式显示分类的业务逻辑代码!接下来介绍一下BeanUtils.copyProperties()这个方法,简单来说:就是一个get+set的过程;复杂点说:直接点进去看源码——>

private static void copyProperties(Object source, Object target, Class<?> editable, String... ignoreProperties) throws BeansException {
    Assert.notNull(source, "Source must not be null");
    Assert.notNull(target, "Target must not be null");
    //获取目标类型对象
    Class<?> actualEditable = target.getClass(); 
    if (editable != null) { //null
        if (!editable.isInstance(target)) {
            throw new IllegalArgumentException("Target class [" + target.getClass().getName() + "] not assignable to Editable class [" + editable.getName() + "]");
        }
 
        actualEditable = editable;
    }
 
    //获取目标对象所有字段属性
    PropertyDescriptor[] targetPds = getPropertyDescriptors(actualEditable); 
    //需要过滤的字段
    List<String> ignoreList = ignoreProperties != null ? Arrays.asList(ignoreProperties) : null; 
    PropertyDescriptor[] var7 = targetPds;
    int var8 = targetPds.length;
 
    for(int var9 = 0; var9 < var8; ++var9) {
        //获取目标字段
        PropertyDescriptor targetPd = var7[var9]; 
        //获取set方法
        Method writeMethod = targetPd.getWriteMethod(); 
        //set方法不为空,过滤字段为空或过滤字段不包括当前遍历字段
        if (writeMethod != null && (ignoreList == null || !ignoreList.contains(targetPd.getName()))) {
            //获取源字段
            PropertyDescriptor sourcePd = getPropertyDescriptor(source.getClass(), targetPd.getName()); 
            if (sourcePd != null) {
                //获取get方法
                Method readMethod = sourcePd.getReadMethod(); 
                //ClassUtils.isAssignable判断是否可转(目标类型是否为源类型父类,或相同)
                if (readMethod != null && ClassUtils.isAssignable(writeMethod.getParameterTypes()[0], readMethod.getReturnType())) {
                    try {
                        //判断是否为public
                        if (!Modifier.isPublic(readMethod.getDeclaringClass().getModifiers())) { 
                             //打破封装
                             readMethod.setAccessible(true); 
                        }
                        //value赋值
                        Object value = readMethod.invoke(source); 
                        if (!Modifier.isPublic(writeMethod.getDeclaringClass().getModifiers())) {
                            writeMethod.setAccessible(true);
                        }
                        //值拷贝
                        writeMethod.invoke(target, value); 
                    } catch (Throwable var15) {
                        throw new FatalBeanException("Could not copy property '" + targetPd.getName() + "' from source to target", var15);
                    }
                }
            }
        }
    }

完结!
我:小弟弟,懂了不
小学弟:懂了懂了!!geigei~ 你好帅啊~
我 :害

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

IT派同学

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

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

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

打赏作者

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

抵扣说明:

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

余额充值