• linkedu视频
  • 平面设计
  • 电脑入门
  • 操作系统
  • 办公应用
  • 电脑硬件
  • 动画设计
  • 3D设计
  • 网页设计
  • CAD设计
  • 影音处理
  • 数据库
  • 程序设计
  • 认证考试
  • 信息管理
  • 信息安全
菜单
linkedu.com
  • 网页制作
  • 数据库
  • 程序设计
  • 操作系统
  • CMS教程
  • 游戏攻略
  • 脚本语言
  • 平面设计
  • 软件教程
  • 网络安全
  • 电脑知识
  • 服务器
  • 视频教程
  • JavaScript
  • ASP.NET
  • PHP
  • 正则表达式
  • AJAX
  • JSP
  • ASP
  • Flex
  • XML
  • 编程技巧
  • Android
  • swift
  • C#教程
  • vb
  • vb.net
  • C语言
  • Java
  • Delphi
  • 易语言
  • vc/mfc
  • 嵌入式开发
  • 游戏开发
  • ios
  • 编程问答
  • 汇编语言
  • 微信小程序
  • 数据结构
  • OpenGL
  • 架构设计
  • qt
  • 微信公众号
您的位置:首页 > 程序设计 >Java > MyBatis的嵌套查询解析

MyBatis的嵌套查询解析

作者:不能说的秘密go 字体:[增加 减小] 来源:互联网 时间:2017-05-28

不能说的秘密go 通过本文主要向大家介绍了mybatis嵌套查询,mybatis if 嵌套,mybatis foreach 嵌套,mybatis 嵌套表,mybatis源码解析等相关知识,希望对您有所帮助,也希望大家支持linkedu.com www.linkedu.com

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 省

 }
</div>

User_orders:

public class User_orders {
 private int id;
 private String name;
 // get set 省
}
</div>

对应的数据库:

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)
</div>

现在查询一个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>
</div>

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

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();
</div>

嵌套语句查询的原理

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

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

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

当要对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
}
</div>

配置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>
</div>

测试:

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();
</div>

嵌套查询的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>
</div>

对应的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 o
  on o.user_id = u.id
  </select>
</div>

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

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

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

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

分享到:QQ空间新浪微博腾讯微博微信百度贴吧QQ好友复制网址打印

您可能想查找下面的文章:

  • MyBatis的嵌套查询解析
  • MyBatis的嵌套查询解析

相关文章

  • 2017-05-28详解JDBC使用
  • 2017-05-28Java操作另一个Java程序使其重启的简单实现
  • 2017-05-28java中栈和队列的实现和API的用法(详解)
  • 2017-05-28Java NIO:浅析IO模型_动力节点Java学院整理
  • 2017-05-28详解Spring Boot实现日志记录 SLF4J
  • 2017-05-28详解SpringBoot多跨域请求的支持(JSONP)
  • 2017-05-28Spring Boot中使用jdbctemplate 操作MYSQL数据库实例
  • 2017-05-28详解spring Boot Cli的配置和使用
  • 2017-05-28SpringBoot加载静态资源的方式
  • 2017-05-28Java字符编码原理(动力节点Java学院整理)

文章分类

  • JavaScript
  • ASP.NET
  • PHP
  • 正则表达式
  • AJAX
  • JSP
  • ASP
  • Flex
  • XML
  • 编程技巧
  • Android
  • swift
  • C#教程
  • vb
  • vb.net
  • C语言
  • Java
  • Delphi
  • 易语言
  • vc/mfc
  • 嵌入式开发
  • 游戏开发
  • ios
  • 编程问答
  • 汇编语言
  • 微信小程序
  • 数据结构
  • OpenGL
  • 架构设计
  • qt
  • 微信公众号

最近更新的内容

    • 超实用的Java快捷键(总结)
    • Java基于正则实现的日期校验功能示例
    • java根据模板动态生成PDF实例
    • java生成缩略图的方法示例
    • JVM教程之Java代码编译和执行的整个过程(二)
    • java用重定向方法从文件中读入或写入数据
    • java中ArrayList与LinkedList对比详情
    • Java常用字符串方法小结
    • java 中HttpClient传输xml字符串实例详解
    • Java自定义注解实现Redis自动缓存的方法

关于我们 - 联系我们 - 免责声明 - 网站地图

©2020-2025 All Rights Reserved. linkedu.com 版权所有