一对多分页的SQL到底应该怎么写?

发布时间:2025-05-14 01:45:26 作者:益华网络 来源:undefined 浏览量(1) 点赞(1)
摘要:前言 MySQL一对多的数据分页是非常常见的需求,比如我们要查询商品和商品的图片信息。但是很多人会在这里遇到分页的误区,得到不正确的结果。今天就来分析并解决这个问题。 2. 问题分析 我们先创建一个简单商品表和对应的商品图片关系表,它们之间是一对多的关系:

前言

MySQL一对多的数据分页是非常常见的需求,比如我们要查询商品和商品的图片信息。但是很多人会在这里遇到分页的误区,得到不正确的结果。今天就来分析并解决这个问题。

2. 问题分析

我们先创建一个简单商品表和对应的商品图片关系表,它们之间是一对多的关系:

然后我分别写入了一些商品和这些商品对应的图片,通过下面的左连接查询可以看出它们之间具有明显的一对多关系:

SELECT P.PRODUCT_ID, P.PROD_NAME, PI.IMAGE_URL FROM PRODUCT_INFO P          LEFT JOIN PRODUCT_IMAGE PI                    ON P.PRODUCT_ID = PI.PRODUCT_ID

按照传统的思维我们的分页语句会这么写:

<resultMap id="ProductDTO" type="cn.felord.mybatis.entity.ProductDTO">     <id property="productId" column="product_id"/>     <result property="prodName" column="prod_name"/>     <collection property="imageUrls"  ofType="string">         <result column="image_url"/>     </collection> </resultMap> <select id="page" resultMap="ProductDTO">     SELECT P.PRODUCT_ID, P.PROD_NAME,PI.IMAGE_URL     FROM PRODUCT_INFO P              LEFT JOIN PRODUCT_IMAGE PI                        ON P.PRODUCT_ID = PI.PRODUCT_ID     LIMIT #{current},#{size} </select>

当我按照预想传入了(0,2)想拿到前两个产品的数据,结果并不是我期望的:

2020-06-21 23:35:54.515 DEBUG 10980 --- [main] c.f.m.mappers.ProductInfoMapper.page     : ==>  Preparing: SELECT P.PRODUCT_ID, P.PROD_NAME,PI.IMAGE_URL FROM PRODUCT_INFO P LEFT JOIN PRODUCT_IMAGE PI ON P.PRODUCT_ID = PI.PRODUCT_ID limit ?,?  2020-06-21 23:35:54.541 DEBUG 10980 --- [main] c.f.m.mappers.ProductInfoMapper.page     : ==> Parameters: 0(Long), 2(Long) 2020-06-21 23:35:54.565 DEBUG 10980 --- [main] c.f.m.mappers.ProductInfoMapper.page     : <==      Total: 2 page = [ProductDTO{productId=1, prodName=杯子, imageUrls=[http://asset.felord.cn/cup1.png, http://asset.felord.cn/cup2.png]}]

我期望的两条数据是杯子和笔记本,但是结果却只有一条。原来当一对多映射时结果集会按照多的一侧进行输出(期望4条数据,实际上会有7条),而前两条展示的只会是杯子的数据(如上图),合并后就只有一条结果了,这样分页就对不上了。那么如何才能达到我们期望的分页效果呢?

3. 正确的方式

正确的思路是应该先对主表进行分页,再关联从表进行查询。

抛开框架,我们的SQL应该先对产品表进行分页查询然后再左关联图片表进行查询:

SELECT P.PRODUCT_ID, P.PROD_NAME, PI.IMAGE_URL FROM (SELECT PRODUCT_ID, PROD_NAME       FROM PRODUCT_INFO       LIMIT #{current},#{size}) P          LEFT JOIN PRODUCT_IMAGE PI                    ON P.PRODUCT_ID = PI.PRODUCT_ID

这种写法的好处就是通用性强一些。但是MyBatis提供了一个相对优雅的路子,思路依然是开头所说的思路。只不过我们需要改造上面的Mybatis XML配置:

<resultMap id="ProductDTO" type="cn.felord.mybatis.entity.ProductDTO">     <id property="productId" column="product_id"/>     <result property="prodName" column="prod_name"/>      <!-- 利用 collection 标签提供的 select 特性 和 column   -->     <collection property="imageUrls" ofType="string" select="selectImagesByProductId" column="product_id"/> </resultMap> <!-- 先查询主表的分页数据    --> <select id="page" resultMap="ProductDTO">     SELECT PRODUCT_ID, PROD_NAME     FROM PRODUCT_INFO     LIMIT #{current},#{size} </select> <!--根据productId 查询对应的图片--> <select id="selectImagesByProductId" resultType="string">     SELECT IMAGE_URL     FROM PRODUCT_IMAGE     WHERE PRODUCT_ID = #{productId} </select>

4. 总结

大部分情况下分页是很容易的,但是一对多还是有一些小小的陷阱的。一旦我们了解了其中的机制,也并不难解决。当然如果你有更好的解决方案可以留言讨论,集思广益。多多关注:码农小胖哥,获取更多开发技巧。

二维码

扫一扫,关注我们

声明:本文由【益华网络】编辑上传发布,转载此文章须经作者同意,并请附上出处【益华网络】及本页链接。如内容、图片有任何版权问题,请联系我们进行处理。

感兴趣吗?

欢迎联系我们,我们愿意为您解答任何有关网站疑难问题!

您身边的【网站建设专家】

搜索千万次不如咨询1次

主营项目:网站建设,手机网站,响应式网站,SEO优化,小程序开发,公众号系统,软件开发等

立即咨询 15368564009
在线客服
嘿,我来帮您!