通用的递归向下查询方法

适用场景

源码:


    /**
     * 从给定的pos中向下加载所有次级数据
     *
     * 这是一个通用的递归查询方法 ,适用业务范围 : 评论,分类,推荐关联
     * 这些业务的表设计都存在嵌套形式 因此可以使用该方法完成通用的递归查询
     *
     * @param poService po的service对象 用于操作mybatis-plus执行数据库查询
     * @param pos 给定的 pos 数据
     * @param vCls vo对象类型
     * @param linkField 次级数据和主数据 的关联关系字段
     * @param customWrapper 用户自定义的wrapper查询条件
     * @return 所有次级数据(包含这次提供的主数据)
     *
     * @param <P>  主级类类对象泛型 (主级类必须实现BasePo接口)
     * @param <V>  次级类对象泛型 (次级类必须实现 RecursionBean接口 , 必须继承自po )
     * @param <M1> 关联关系字段泛型
     */
    @SuppressWarnings("unchecked")
    public static<P ,V, M1> List< V > uploadSubsForPos( IService<P> poService,
                                                        List<P> pos,
                                                        Class<V> vCls,
                                                        SFunction<P, M1> linkField,
                                                        Consumer<LambdaQueryWrapper<P>> customWrapper) {

        //---------------------------------------------构建查询次级数据的条件---------------------------------------------

        //查询次级数据时 条件必定需要主数据所有的主键id 那么把所有主数据的id提取出来
        List<M1> ids =( List<M1> ) ListUtils.collectIds( pos );

        //开始构建查询条件 使用提供的 linkField 和ids 构建初始查询条件
        LambdaQueryWrapper<P> wrapper = MybatisPlusUtils.in( ids , linkField );

        //持续补充用户自定义的条件...
        if(null!=wrapper && null!= customWrapper){
            wrapper = wrapper.and( customWrapper );
        }

        //-------------------------------------------------查询次级数据--------------------------------------------------

        //查询 次级数据
        List<P> allSubs = MybatisPlusUtils.safeList(poService,wrapper);

        //根据提供的linkField 给次级数据分组 方便主数据获取属于自己的次级数据
        Map<M1, List<P>> subMap = ListUtils.groupingBy( allSubs, linkField );

        //------------------------------------------------开始封装主数据-------------------------------------------------

        //pos转vos (主数据类型是没有subs字段的 因此转为vo 才可以封装subs字段)
        List<V> vos = BeanUtils.copyList( pos, vCls );

        //todo 此处的泛型转换相当于 PO --> VO --> VO实现的RecursionBean 为什么可以这么转换?

        //开始遍历封装
        for (V v : vos) {
            if(v instanceof RecursionBean){

                //将这个主数据转为 RecursionBean类 才可以设置subs
                RecursionBean<V> vo = (RecursionBean<V>) v;
                M1 id = (M1) vo.getId();

                //获取这个主数据的次级数据
                List<P> subs = subMap.getOrDefault( id ,new ArrayList<>() );

                //给这个主数据 设置它的次级数据 (此处开始递归... 会封装次级数据的次级数据的次级数据的....)
                vo.setSubs(  uploadSubsForPos( poService , subs ,vCls, linkField, customWrapper ) );

            }else {
                //todo 未正确继承异常 后续统一异常信息提示
                throw new RuntimeException("");
            }
        }
        return vos;
    }

1

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值