站在巨人的肩膀上
 
 http://crocutax.com/blog/mybatis-one-to-many-nestes-query-and-page-query

Mybatis一对多嵌套查询和分页

需求:根据分类ID查询分类下所属的商品集合,每个商品又有一个图片集合。

类似的需求有很多,比如经典的一个用户有N个角色,一个角色有N个权限,那么通过用户的id来查询角色和权限数据等等。

至于分页插件,无论是Mybatis-PageHelper还是Mybatis-Plus都可以辅助,这里主要记录不同查询方式对分页的影响。

先展示结果:

Copy{
"code": 0,
"msg": "success",
"data": {
"total": 9,
"size": 2,
"pages": 5,
"current": 1,
"records": [
{
"id": 1,
"code": "1410854032",
"name": "Esmeralda Kilback",
"categoryId": "1",
"originPrice": 359,
"price": 103,
"sales": 299,
"commentCount": 0,
"freight": 1,
"detail": "这里是商品详情",
"createdAt": "2018-04-09 18:52:05",
"updatedAt": "2018-04-24 23:41:49",
"images": [
{
"id": 40,
"productId": "1",
"link": "uploads/product/201804/18/78a6e4e4d73bfc64b7aef88a90e7f192.png",
"createdAt": "2018-04-09 18:52:05",
"updatedAt": "2018-04-18 16:37:09"
},
{
"id": 41,
"productId": "1",
"link": "uploads/product/201804/18/fffdccaa36a8475ed3d2c71c2f43cb86.png",
"createdAt": "2018-04-09 18:52:05",
"updatedAt": "2018-04-18 16:37:09"
},
{
"id": 301,
"productId": "1",
"link": "uploads/product/201804/18/68b18cbcb090a94123abd9d729528370.png",
"createdAt": "2018-04-18 16:35:56",
"updatedAt": "2018-04-18 16:35:56"
}
]
},
{
"id": 8,
"code": "1925117917",
"name": "Edgardo Osinski",
"categoryId": "1",
"originPrice": 389,
"price": 154,
"sales": 199,
"commentCount": 0,
"freight": 14,
"detail": "这里是商品详情...5052 Kyler Walk Suite 921",
"createdAt": "2018-04-09 18:52:05",
"updatedAt": "2018-04-09 18:52:05",
"images": [
{
"id": 58,
"productId": "8",
"link": "uploads/default.png",
"createdAt": "2018-04-09 18:52:05",
"updatedAt": "2018-04-09 18:52:05"
},
{
"id": 59,
"productId": "8",
"link": "uploads/default2.png",
"createdAt": "2018-04-09 18:52:05",
"updatedAt": "2018-04-09 18:52:05"
},
{
"id": 60,
"productId": "8",
"link": "uploads/default3.png",
"createdAt": "2018-04-09 18:52:05",
"updatedAt": "2018-04-09 18:52:05"
}
]
}
]
}
}

定义模型

Product用于数据库映射,为了保持其简洁,其他的二次封装不在Product里进行,而用继承的方式。

定义模型ProductVo

Copy@Data
public class ProductVo extends Product { private List<ProductImage> images;
}

方式1:结果查询

  1. 在ProductsMapper.xml中定义select语句,一次性将关联数据全部查询出来,然后进行结果映射
Copy    <select id="selectProductsBycategoryId"  resultMap="productsListMap">
select
p.id,
p.name,
p.code,
...
i.id images_id,
i.product_id images_product_id,
...
from products p
inner join product_images i on p.id = i.product_id
where p.category_id = #{id}
</select>
  1. 定义productsListMap结果映射
Copy    <resultMap id="productsListMap" type="com.longke.mallb2c.entity.vo.ProductVo" extends="BaseResultMap">-->
<collection property="images" columnPrefix="images_" resultMap="com.longke.mallb2c.mapper.ProductImagesMapper.BaseResultMap"/>
</resultMap>

注意:

  • property就是在ProductVo中定义的商品图片集合images字段
  • 用到了columnPrefix列前缀,只是别名前缀,跟数据库内的字段无关,这个images_别名前缀跟select中定义别名时要保持一致。
  • 用到了extends继承已有的BaseResultMap,不用在这里再重新写Product表的每个字段的映射了。mapper.xml自动生成工具都会帮我们生成这个BaseResultMap,直接继承即可。
  • collection用于一对多查询,查询的结果映射直接复用ProductImagesMapper中定义的BaseResultMap

总结

优点

  • 一次性查询,集中映射,简单,效率

缺点

  • 会将collection中查询到的条数作为分页的约束条件,导致分页数据不准确。

比如想查page=1,limit=10的数据,本来期望的是查询出10个商品,然后这10个商品分别再嵌套查询出自己的商品图片集合。但是会发现,可能商品只有两三个,每个下面都带了自己的商品图片集合。

原因:

先通过表连接把表记录关联进来了,如果有3个商品,关联图片表之后每个商品有4条图片记录,那么其实这时候虽然只有三个商品,但是这个内存中的临时表已经有12条记录了,在语句的最后加上 limit 0,10,其实分页的时候分的是这12条记录。最终就会导致最终的映射结果只出现了3个商品,而非我们期望的10个商品。

方式2:嵌套查询

  1. 在ProductsMapper.xml中定义select语句
Copy   <select id="selectProductsBycategoryId"  resultMap="productsListMap">
select <include refid="Base_Column_List"/>
from products
where category_id = #{id}
</select>
  1. 定义productsListMap结果映射
Copy    <resultMap id="productsListMap" type="com.longke.mallb2c.entity.vo.ProductVo" extends="BaseResultMap">
<collection property="images" ofType="com.longke.mallb2c.entity.ProductImage"
column="{productId=id}" select="com.longke.mallb2c.mapper.ProductImagesMapper.selectByProductId">
</collection>
</resultMap>

注意:

  • column是参数传递,即将Product的哪个属性传递给嵌套的查询语句,{productId=id}代表将Product的id属性传递给参数productId
  • select直接使用ProductImagesMapper中定义的select语句
  1. ProductImagesMapper定义selectByProductId查询语句
Copy    <select id="selectByProductId" resultMap="BaseResultMap">
SELECT <include refid="Base_Column_List"/>
from product_images
where product_id = #{productId}
</select>

总结

优点

  • 准确分页

缺点

  • 没有解决N+1的问题,多条SQL查询语句,效率太差

Mybatis分页中遇到的坑2的更多相关文章

  1. Mybatis分页中遇到的坑3

    Mybatis Mapper.xml 配置文件中 resultMap 节点的源码解析   相关文章 Mybatis 解析配置文件的源码解析 Mybatis 类型转换源码分析 Mybatis 数据源和数 ...

  2. Mysql系列八:Mycat和Sharding-jdbc的区别、Mycat分片join、Mycat分页中的坑、Mycat注解、Catlet使用

    一.Mycat和Sharding-jdbc的区别 1)mycat是一个中间件的第三方应用,sharding-jdbc是一个jar包 2)使用mycat时不需要改代码,而使用sharding-jdbc时 ...

  3. Mybatis分页和Spring的集成

    写了一个Mybatis分页控件,在这记录一下使用方式. 在Maven中加入依赖: ? 1 2 3 4 5 6 7 8 9 <dependencies>   ...     <depe ...

  4. mybatis分页插件以及懒加载

    1.   延迟加载 延迟加载的意义在于,虽然是关联查询,但不是及时将关联的数据查询出来,而且在需要的时候进行查询. 开启延迟加载: <setting name="lazyLoading ...

  5. Mybatis分页插件PageHelper正确的用法(网上有2篇不够科学的文章)

    今天下午在Mybatis项目中.实现分页.由于我是后加入项目中的,Leader用的是PageHelper这个组件.可是我在实际使用的过程中遇到了2个大问题. 1.p=2#comments" ...

  6. Mybatis分页插件PageHelper正确的使用方法(网上有2篇不够科学的文章)

    今天下午在Mybatis项目中,实现分页.因为我是后加入项目中的,Leader用的是PageHelper这个组件,但是我在实际使用的过程中遇到了2个大问题. 1.http://www.oschina. ...

  7. Mybatis分页插件PageHelper的配置和使用方法

     Mybatis分页插件PageHelper的配置和使用方法 前言 在web开发过程中涉及到表格时,例如dataTable,就会产生分页的需求,通常我们将分页方式分为两种:前端分页和后端分页. 前端分 ...

  8. Mybatis分页插件PageHelper使用

    一. Mybatis分页插件PageHelper使用  1.不使用插件如何分页: 使用mybatis实现: 1)接口: List<Student> selectStudent(Map< ...

  9. mybatis分页练手

    最近碰到个需求,要做个透明的mybatis分页功能,描述如下:目标:搜索列表的Controller action要和原先保持一样,并且返回的json需要有分页信息,如: @ResponseBody @ ...

随机推荐

  1. xmanager 5图文使用教程

    1.Xconfig xconfig是linux下X Window环境中用于配制的一个工具,和menuconfig相似,但用法更友好方便. 当你创建一个会话,会话分配一个默认的配置文件.Xmanager ...

  2. dokcer3

    安装好的文件位置: /usr/sbin/nginx:主程序 /etc/nginx:存放配置文件 /usr/share/nginx:存放静态文件 /var/log/nginx:存放日志 其实从上面的根目 ...

  3. Linux下直接读写物理地址内存

    虚拟 转 物理地址  virt_to_phys( *addr );物理 转 虚拟地址  phys_to_virt( *addr ); 如: unsigned long pProtectVA; phys ...

  4. hihocoder #1122 二分图二•二分图最大匹配之匈牙利算法(*【模板】应用 )

    梳理整个算法: 1. 依次枚举每一个点i: 2. 若点i尚未匹配,则以此点为起点查询一次交错路径. 最后即可得到最大匹配数. 在这个基础上仍然有两个可以优化的地方: 1.对于点的枚举:当我们枚举了所有 ...

  5. yum的配置文件yum.conf详解

    说明:经过网上抄袭和自己的总结加实验,非常详细,可留作参考. yum的配置一般有两种方式:   一种是直接配置/etc目录下的yum.conf文件, 另外一种是在/etc/yum.repos.d目录下 ...

  6. PICT实现组合测试用例

    成功安装后,在命令行中输入命令pict: 可以看到pict命令的一些选项: /o:N   组合数,默认值为2,即pict生成的测试用例集中每条测试数据会有两个值与其他测试集是不同的: /d:C   值 ...

  7. 单元测试JUnit4 Ctrl + Shift + T

    单元测试 public class Calculator { public int result = 0; public int add(int operandl, int operand2) { r ...

  8. PowerDesigner 连接 MySQL 并生成逆向工程图

    1 配置环境变量 Tools → General Options → Variables   配置 JVM 变量 General Options 在最后,要一直往下拉才会看到. 注意: PowerDe ...

  9. 如何应用 AutoIt 修改本机的防火墙配置?(开启,关闭防火墙,添加程序信任到防火墙)

    以前,公司的实施人员配置好项目之后,不同的机器之间经常性的无法建立链接,后来发现是防火墙的设置.虽然是个小问题,但是经常性的忘记这个配置. 现在,我决定把对防火墙的设置,加入到我给实施人员的配置工具中 ...

  10. Quicklz压缩算法

    以前对压缩算法一无所知,只是知道哈弗曼编码能做这种事情,但是感觉这样的方法奇慢无比.昨天下午看了下号称世界上最快的压缩算法Quicklz,对压缩的基本思路有了一定的了解.一般的压缩程序的要求读入文件之 ...