mybatis-一对多分页查询

  • 问题
    在用mybatis做一对多查询时候,常用collection配合完成结果查询。在不涉及分页查询情况下,查询结果是没有问题的。但当涉及分页查询时,就会出现问题,即结果总数量total多于实际数量。

  • 演示示例如下

  • 实体类

@Data
public class JudgePicPointSatatusResVo {
    private Integer id;
    private String judgePicPoint;
    private String productNum;
    private String judgePicStation;
    private String stationName;
    private List<String> stationNameList;
    private Integer status;
}
  • Dao层
List<JudgePicPointSatatusResVo> getJudgePicPointListByJudgePicStationId(JudgePicPointSatatusReqVo judgePicPointSatatusReqVo);
  • 查询
    <resultMap id="judgePicPointListMap" type="com.qxmz.vo.judgePic.JudgePicPointSatatusResVo">
        <id column="id" property="id"/>
        <result column="judgePicPoint" property="judgePicPoint"/>
        <result column="productNum" property="productNum"/>
        <result column="judgePicStation" property="judgePicStation"/>
        <result column="status" property="status"/>
        <collection property="stationNameList" ofType="String">
            <id column="stationName"/>
        </collection>
    </resultMap>
    <select id="getJudgePicPointListByJudgePicStationId" parameterType="com.qxmz.vo.judgePic.JudgePicPointSatatusReqVo"
            resultMap="judgePicPointListMap">
        SELECT cj.`id`,cj.`productNum`,cj.`status`, jpp.`name` judgePicPoint,jps.`name`
        judgePicStation,st.station_name stationName FROM client_judgePicPoint cj
        LEFT JOIN judge_pic_point jpp ON cj.`judgePicPointId`=jpp.`id`
        LEFT JOIN judge_pic_station jps ON jpp.`judgePicStationId`=jps.`id`
        LEFT JOIN station st ON jps.`id`=st.`judgePicStationId`
        <where>
            cj.isAdd=1
            <if test="judgePicStationIdList!=null">
                and jps.id in
                <foreach item="judgePicStationId" collection="judgePicStationIdList" open="(" separator="," close=")">
                    #{judgePicStationId}
                </foreach>
            </if>
            <if test="status != null and status != 0">
                and cj.status = #{status}
            </if>
        </where>
    </select>
  • 结果
{
    "code": 200,
    "message": "success",
    "data": {
        "total": 2,
        "rows": [
            {
                "id": 2,
                "judgePicPoint": "判图点3",
                "productNum": "002",
                "judgePicStation": "判图站B",
                "stationName": "郑州西站、南阳寨站",
                "stationNameList": [
                    "郑州西站",
                    "南阳寨站"
                ],
                "status": 2
            }
        ]
    }
}

从结果中可以看到,实际只查询出来一条数据,但总数却是2。这就是一对多分页查询会出现的问题。原因也很简单,以上面数据为例。在mysql客户端执行一对多SQL命令时,结果就是两条。一对多情况下,mybatis会自动将结果装备到collection中,但是在分页情况下,mybatis会认为查询的两台数据就是总的数据。

  • 解决方法
    采用父子查询来实现,具体代码如下:
  • 父查询
    <resultMap id="judgePicPointListMap" type="com.qxmz.vo.judgePic.JudgePicPointSatatusResVo">
        <id column="id" property="id"/>
        <result column="judgePicPoint" property="judgePicPoint"/>
        <result column="productNum" property="productNum"/>
        <result column="judgePicStation" property="judgePicStation"/>
        <result column="status" property="status"/>
        <collection property="stationNameList" ofType="String" column="judgePicStationId"
                    select="selectStationNameByJudgePicStationId">
        </collection>
    </resultMap>
    <select id="getJudgePicPointListByJudgePicStationId" parameterType="com.qxmz.vo.judgePic.JudgePicPointSatatusReqVo"
            resultMap="judgePicPointListMap">
        SELECT cj.`id`,cj.`productNum`,cj.`status`, jpp.`name` judgePicPoint,jps.`name`
        judgePicStation,jps.id judgePicStationId FROM client_judgePicPoint cj
        LEFT JOIN judge_pic_point jpp ON cj.`judgePicPointId`=jpp.`id`
        LEFT JOIN judge_pic_station jps ON jpp.`judgePicStationId`=jps.`id`
        <where>
            cj.isAdd=1
            <if test="judgePicStationIdList!=null">
                and jps.id in
                <foreach item="judgePicStationId" collection="judgePicStationIdList" open="(" separator="," close=")">
                    #{judgePicStationId}
                </foreach>
            </if>
            <if test="status != null and status != 0">
                and cj.status = #{status}
            </if>
        </where>
    </select>
  • 子查询
<select id="selectStationNameByJudgePicStationId" resultType="String">
        SELECT  station_name FROM station WHERE judgePicStationId=#{judgePicStationId}
    </select>
  • 正确结果
{
    "code": 200,
    "message": "success",
    "data": {
        "total": 1,
        "rows": [
            {
                "id": 2,
                "judgePicPoint": "判图点3",
                "productNum": "002",
                "judgePicStation": "判图站B",
                "stationName": "郑州西站、南阳寨站",
                "stationNameList": [
                    "郑州西站",
                    "南阳寨站"
                ],
                "status": 2
            }
        ]
    }
}
### MyBatis-Plus 使用 JOIN 进行分查询 为了提升多表连接查询的能力,MyBatis-Plus Join 工具提供了 `MPJLambdaWrapper` 和 `MPJQueryWrapper` 类用于执行复杂的多表操作[^1]。这些工具不仅简化了 SQL 的编写过程,还增强了对多表间关系的支持。 #### 实现分查询的具体方式 通过使用 `MPJLambdaWrapper` 可以方便地指定联结条件以及设定额外的过滤规则,并且能够自然地集成到现有的分机制当中[^2]。下面是一个简单的例子展示如何结合分插件来获取带有用户订单详情的结果集: ```java // 导入必要的包 import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import org.example.entity.User; // 用户实体类 import org.example.mapper.UserMapper; public class UserService { private final UserMapper userMapper; public IPage<UserOrderDTO> getUserOrders(Integer userId, Page page) { MPJLambdaWrapper<User> wrapper = new MPJLambdaWrapper<>(); // 设置主表为User并加入与Order表之间的JOIN语句 wrapper.select(User::getId, User::getName) .selectAs(Order::getOrderId, "order_id") .leftJoin("orders o on u.id = o.user_id") // 假设存在名为orders的订单表 // 添加其他筛选条件(如果有的话) if (userId != null && userId > 0){ wrapper.eq(User::getId, userId); } // 执行带分参数的查询 return userMapper.selectJoinPage(page, wrapper, UserOrderDTO.class); } } ``` 此代码片段展示了怎样创建一个服务层方法来进行基于用户的订单记录检索,其中包含了基本的分逻辑和跨表字段的选择。注意这里假设有一个映射文件已经配置好了相应的 XML 或者动态 SQL 来支持这种类型的联合查询
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值