Mybatis表现关联关系比hibernate简单,没有分那么细致one-to-many、many-to-one、one-to-one。而是只有两种association(一)、collection(多),表现很简洁。下面通过一个实例,来展示一下Mybatis对于常见的一对多和多对一关系复杂映射是怎样处理的。

以最简单的用户表订单表这个最简单的一对多做示例:

对应的JavaBean:

User:

public class User {private int id;private String name;private Double age;private List<User_orders> orders;// get set 省}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

User_orders:

public class User_orders {private int id;private String name;// get set 省
}
  • 1
  • 2
  • 3
  • 4
  • 5

对应的数据库:

mysql> desc user;
+-------+-------------+------+-----+---------+----------------+
| Field | Type        | Null | Key | Default | Extra          |
+-------+-------------+------+-----+---------+----------------+
| id    | int(11)     | NO   | PRI | NULL    | auto_increment |
| name  | varchar(20) | NO   |     | NULL    |                |
| age   | double      | YES  |     | NULL    |                |
+-------+-------------+------+-----+---------+----------------+
3 rows in set (0.00 sec)mysql> desc user_orders;
+---------+-------------+------+-----+---------+----------------+
| Field   | Type        | Null | Key | Default | Extra          |
+---------+-------------+------+-----+---------+----------------+
| id      | int(11)     | NO   | PRI | NULL    | auto_increment |
| name    | varchar(20) | NO   |     | NULL    |                |
| user_id | int(5)      | YES  | MUL | NULL    |                |
+---------+-------------+------+-----+---------+----------------+
3 rows in set (0.00 sec)
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20

现在查询一个user的id查询出所有信息.如果不考虑关联查询,我们会先根据user的id在user表中查询出name,age然后设置给User类的时候,再根据该user的id在user_orders表中查询出所有订单并设置给User类。这样的话,在底层最起码调用两次查询语句,得到需要的信息,然后再组装User对象。

嵌套语句查询

mybatis提供了一种机制,叫做嵌套语句查询,可以大大简化上述的操作,加入配置及代码如下:

<resultMap type="domain.User" id="user"><id column="id" property="id"/><result column="age" property="age"/><collection column="id" property="orders" ofType="domain.User_orders"select="selectOrderByUser"> <id column="id" property="id"/><result column="name" property="name"/></collection>
</resultMap>
<select id="selectOrderByUser" parameterType="integer" resultType="domain.User_orders">select id,name from user_orders where user_id = #{id}
</select><select id="findById" resultMap="user" parameterType="integer">select * from user where id = #{id}</select>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17

测试(可以成功查询到所有信息):

String config = "sqlMapConfig.xml";
InputStream inputStream = Resources.getResourceAsStream(config);
sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
SqlSession session = sqlSessionFactory.openSession();
// 执行在bean配置文件中定义的sql语句
User user = session.selectOne("UserMapper.findById", 1);
//一句即可获取到复杂的User对象。
System.out.println(user);
session.commit();
session.close();
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10

嵌套语句查询的原理

在上面的代码中,Mybatis会执行以下流程:

  • 1.先执行 findById 对应的语句从User表里获取到ResultSet结果集;

  • 2.取出ResultSet下一条有效记录,然后根据resultMap定义的映射规格,通过这条记录的数据来构建对应的一个User 对象。

    1. 当要对User中的orders属性进行赋值的时候,发现有一个关联的查询,此时Mybatis会先执行这个select查询语句,得到返回的结果,将结果设置到user的orders属性上

这种关联的嵌套查询,有一个非常好的作用就是:可以重用select语句,通过简单的select语句之间的组合来构造复杂的对象。想如上的两个select完全可以独立使用。

嵌套查询的多对一 
上面的关联查询查询其实是对于一对多的查询,即从user中查出user_order的信息。 
现在从user_order中查user的信息. 
在User_order表中增加字段user:

public class User_orders {private int id;private String name;private User user;//xxx
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

配置select:

<resultMap type="domain.User_orders" id="user_order"><id column="id" property="id"/><result column="name" property="name"/><association property="user" column="user_id" javaType="domain.User" select="selectUserByOrderId"><id column="id" property="id"/><result column="age" property="age"/></association>
</resultMap><select id="selectUserByOrderId" parameterType="INTEGER" resultType="domain.User">select id,age from user where id = #{id}</select><select id="findOne" resultMap="user_order" parameterType="integer">select * from  user_orders where id=#{id}</select>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16

测试:

SqlSession session = sqlSessionFactory.openSession();// 执行在bean配置文件中定义的sql语句User_orders user_orders= session.selectOne("User_ordersMapper.findOne", 1);System.out.println(user_orders);//查询到了user_order对应的user的信息session.commit();session.close();
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

嵌套查询的N+1问题

尽管嵌套查询大量的简化了存在关联关系的查询,但它的弊端也比较明显:即所谓的N+1问题。关联的嵌套查询显示得到一个结果集,然后根据这个结果集的每一条记录进行关联查询。

现在假设嵌套查询就一个(即resultMap 内部就一个association标签),现查询的结果集返回条数为N,那么关联查询语句将会被执行N次,加上自身返回结果集查询1次,共需要访问数据库N+1次。如果N比较大的话,这样的数据库访问消耗是非常大的!所以使用这种嵌套语句查询的使用者一定要考虑慎重考虑,确保N值不会很大。

以上面一对多(根据user的id查询order)的例子为例,select 语句本身会返回user条数为1 的结果集,由于它存在有1条关联的语句查询,它需要共访问数据库 1*(1+1)=2次数据库。

嵌套结果查询

嵌套语句的查询会导致数据库访问次数不定,进而有可能影响到性能。Mybatis还支持一种嵌套结果的查询:即对于一对多,多对多,多对一的情况的查询,Mybatis通过联合查询,将结果从数据库内一次性查出来,然后根据其一对多,多对一,多对多的关系和ResultMap中的配置,进行结果的转换,构建需要的对象。

重新定义User的结果映射 resultMap

<resultMap type="domain.User" id="user_auto">
<id column="id" property="id"/><result column="age" property="age"/><collection column="id" property="orders" ofType="domain.User_orders"> <id column="order_id" property="id"/><result column="name" property="name"/></collection>
</resultMap>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

对应的sql语句如下:

    <select id="findAuth" resultMap="user_auto">select u.id,u.age,o.id as order_id ,o.name,o.user_id as user_id from user u left outer join user_orders oon o.user_id = u.id</select>
  • 1
  • 2
  • 3
  • 4

嵌套结果查询的执行步骤:

  • 1.根据表的对应关系,进行join操作,获取到结果集;

    1. 根据结果集的信息和user 的resultMap定义信息,对返回的结果集在内存中进行组装、赋值,构造User;
    1. 返回构造出来的结果List 结果。

对于关联的结果查询,如果是多对一的关系,则通过形如 <association property="user" column="user_id" javaType="domain.User" >进行配置,Mybatis会通过column属性对应的user_id 值去从内存中取数据,并且封装成User_order对象;

如果是一对多的关系,就如User和User_order之间的关系,通过形如 <collection column="id" property="orders" ofType="domain.User_orders">进行配置,MyBatis通过 id去内存中取User_orders对象,封装成List;

对于关联结果的查询,只需要查询数据库一次,然后对结果的整合和组装全部放在了内存中。

以上是通过查询User表所有信息来演示了一对多和多对一的映射对象处理。

MyBatis嵌套查询解析相关推荐

  1. mybatis嵌套查询和嵌套结果有什么区别_Java面试专题之九:Mybatis面试5个大概率被问到的问题...

    1.为什么说 Mybatis 是半自动 ORM 映射工具?它与全自动的区别在哪里? Hibernate 属于全自动 ORM 映射工具,使用 Hibernate 查询关联对象或者关联集合对象时,可以根据 ...

  2. mybatis -- 嵌套查询

    目录 环境准备 原理解析 源码分析 环境准备 原理解析 首先创建主查询的RowKey, 根据主查询的数据库值 读取暂存区, 判断根据key是否可以获取到值, 如果获取到就不需要创建对象,否则创建对象 ...

  3. Mybatis嵌套查询与嵌套结果

    一对多关系:一是用户,多是订单 实体类User public class User {private Integer id;private String name;private Integer ag ...

  4. mybatis嵌套查询

    类中包含自己 @Data public class SubjectVo {private String id;private int sort;private String title;private ...

  5. 2.4.3 Mybatis 高级查询, 复杂映射, 返回主键, 动态SQL if, set, foreach, 核心配置文件深入,plugins标签, 多表查询, 嵌套查询

    目录 Mybatis 复杂映射&配置文件深入 一 Mybatis高级查询 1.1 ResutlMap属性 1.2 多条件查询(三种) 1.3 模糊查询 二 Mybatis映射文件深入 2.1 ...

  6. MyBatis的关联映射之 一对一(嵌套查询/嵌套结果)

    关联映射概述 在实际的开发中,对数据库的操作常常会涉及多张表,这在面向对象中就涉及了对象与对象之间的关联关系 针对多表之间的操作, MyBatis 提供了关联映射,通过关联映射就可以很好地处理对象与对 ...

  7. mybatis collection column 嵌套查询

    mybatis  嵌套查询 使用可行案例 collection定义关联集合类型的属性的封装规则 ofType:指定集合里面元素的类型 collection中的column属性是数据库中的列名,或着是列 ...

  8. mybatis复杂查询环境 多对一的处理 按照结果嵌套处理和按照查询嵌套处理

    sql插入表student的语句: insert into student (id, name, tid) values ('9', '梁梁','2'); insert into student (i ...

  9. mybatis mysql查询树形结构_MyBatis collection 集合嵌套查询树形节点

    原标题:MyBatis collection 集合嵌套查询树形节点 MyBatis collection 集合 MyBatis 是数据持久层框架,支持定制化 SQL.存储过程以及高级映射.尤其强大在于 ...

最新文章

  1. 从一道面试题谈谈一线大厂码农应该具备的基本能力
  2. python财务预算分析_财码Python管理会计小实验—营运管理之多维度盈利能力分析...
  3. 使用之location和rewrite用法
  4. android 设备名称_如何更改您的Android TV的设备名称
  5. Chrome OS 70 发布:这是安卓的私生子吗?
  6. linux a8启动过程,51CTO博客-专业IT技术博客创作平台-技术成就梦想
  7. Codeforces 61B【怪在读题】
  8. HDOJ 1754 I Hate It 解题报告
  9. 特洛伊木马与计算机病毒有什么区别,特洛伊木马Vs病毒Vs蠕虫, 有什么区别?...
  10. 什么是JavaSE,写给第一次接触Java的人
  11. 云原生分布式数据库和数据仓库崛起背后的原因
  12. 神舟笔记本风扇偏移调多少_神舟战神怎么调节风扇我们该办正事了
  13. 手推公式--马氏距离
  14. 大学生HTML期末作业网页:使用DIV+CSS技术制作一个简单的小说网站 (3个页面 登录+注册+首页 )
  15. OSPF协议的四种网络类型
  16. http库三剑客:httpx
  17. Linux操作系统的设备独立性,Linux的设备管理
  18. ubuntu20下Cuckoo SandBox安装教程--大踩坑版(一)
  19. 全球及中国冬装外套行业市场需求及投资前景调研报告2022-2027年
  20. 什么软件可以文字转语音?快把这些软件收好

热门文章

  1. 机器学习之线性回归(python)
  2. oracle 存储过程 db,oracle数据库的存储过程是什么?
  3. linux怎么删web应用程序错误,Ubuntu 20.04将删除Amazon Web应用程序,但用户可另行安装...
  4. as和java什么关系_深入理解happens-before和as-if-serial语义
  5. matplotlib 横坐标少了一个点_收藏起来!比 matplotlib 效率高十倍的数据可视化神器!...
  6. object detection之Win10配置
  7. .NET 面向对象基础
  8. rocket-console控制台安装
  9. 《Iterative-GAN》的算法伪代码整理
  10. 数据访问增删改查之单项查询