Mybatis源码分析--关联表查询及延迟加载(一)
Mybatis提供了关联查询映射的功能。
一、一对一关联
实体类如下:
Users表对应的实体类:
public class User {
private int id;
private String names;
private int age;
private Teacher teacher;
......getXxx和setXxx方法
}
Teacher表对应的实体类:
public class Teacher {
private int id;
private String name;
.....getXxx和setXxx方法
}
Mapper文件中的配置如下:
<mapper namespace="com.tianjunwei.lazy.entity.Users">
<select id="getUser" parameterType="int" resultMap="user">
select * from users,tbl_teacher where users.id=#{id} and users.teacher_id=tbl_teacher.id
</select>
<resultMap type="com.tianjunwei.lazy.entity.User" id="user" >
<id column="id" property="id" javaType="int" jdbcType="INTEGER"></id>
<result column="name" property="names" javaType="string" jdbcType="VARCHAR"/>
<result column="age" property="age" javaType="int" jdbcType="INTEGER"/>
<association property="teacher" javaType="com.tianjunwei.lazy.entity.Teacher"> <!-- fetchType="lazy" -->
<id property="id" column="id"/>
<result property="name" column="name"/>
</association>
</resultMap>
</mapper>
Main函数如下:
public class LazyUser {
public static void main(String [] args){
//mybatis的配置文件
String resource = "learn/mybatis-config.xml";
InputStream is = LazyUser.class.getClassLoader().getResourceAsStream(resource);
//构建sqlSession的工厂
SqlSessionFactory sessionFactory = new SqlSessionFactoryBuilder().build(is);
SqlSession session = sessionFactory.openSession();
String statement = "com.tianjunwei.lazy.entity.Users.getUser";//映射sql的标识字符串
//执行查询返回一个唯一user对象的sql
User user = session.selectOne(statement, 1);
session.commit(true);
System.out.println(user.getNames());
Teacher teacher = user.getTeacher();
System.err.println(teacher.getName());
}
}
这样就可以通过查询获得一对一的User和Teacher的关联数据。
虽然上面的示例帮助我们查询到了User和Teacher的关联数据,但有时候可能我们并不需要关联数据,并且我们是使用一条sql语句通过两个表的联合查询来获得数据的。Mybatis给我们提供了延迟加载的机制,就是当我们不需要Teacher的数据时,并不会给我们查询,当需要Teacher数据时再进行查询,这样有一点不好的地方就是不是一次查询数据,需要多次连接数据库来获取数据。
1、首选需要在Mybatis的配置文件中添加如下配置:
<!-- 打开延迟加载的开关 --> <setting name="lazyLoadingEnabled" value="true" /> <!-- 将积极加载改为消息加载即按需加载 --> <setting name="aggressiveLazyLoading" value="false"/>
2、修改Mapping文件中的内容:
其实就是执行两次sql语句分别获取Users和Teacher中的值
<mapper namespace="com.tianjunwei.lazy.entity.User">
<select id="getById" parameterType="int" resultMap="user">
select * from users where users.id=#{id}
</select>
<select id="getTeacher" resultMap="teacher">
select * from tbl_teacher where id=#{id}
</select>
<resultMap type="com.tianjunwei.lazy.entity.Teacher" id="teacher" >
<id column="id" property="id" javaType="int" jdbcType="INTEGER"></id>
<result column="name" property="name" javaType="string" jdbcType="VARCHAR"/>
</resultMap>
<resultMap type="com.tianjunwei.lazy.entity.User" id="user" >
<id column="id" property="id" javaType="int" jdbcType="INTEGER"></id>
<result column="name" property="names" javaType="string" jdbcType="VARCHAR"/>
<result column="age" property="age" javaType="int" jdbcType="INTEGER"/>
<association property="teacher" javaType="com.tianjunwei.lazy.entity.Teacher" select="com.tianjunwei.lazy.entity.User.getTeacher" column="id"> <!-- fetchType="lazy" -->
<id property="id" column="id"/>
<result property="name" column="name"/>
</association>
</resultMap>
</mapper>
Main函数中的方法还是和原来的一样
public class LazyMain {
public static void main(String [] args){
//mybatis的配置文件
String resource = "learn/mybatis-config.xml";
InputStream is = LazyMain.class.getClassLoader().getResourceAsStream(resource);
//构建sqlSession的工厂
SqlSessionFactory sessionFactory = new SqlSessionFactoryBuilder().build(is);
SqlSession session = sessionFactory.openSession();
String statement = "com.tianjunwei.lazy.entity.User.getById";//映射sql的标识字符串
//执行查询返回一个唯一user对象的sql
User user = session.selectOne(statement, 1);
session.commit(true);
System.out.println(user.getNames());
Teacher teacher = user.getTeacher();
System.err.println(teacher.getName());
}
}
执行过程:从下图你可以看到是执行了两次Sql语句。
当把下面代码注释掉之后,就是我们不需要teacher的数据时
//Teacher teacher = user.getTeacher(); //System.err.println(teacher.getName());
执行结果如下图,就不会执行teacher相关的sql语句,这样就实现了延迟加载的功能。
以上介绍的是一对一的情况,Mapping文件中使用的是association标签
二、一对多关联
我们介绍一下一对多情况
Teacher的实体定义如下:
public class Teacher {
private int id;
private String name;
private List<User> users;
.....getXxx和setXxx方法....
}
Mapper文件中的定义如下,一对多使用的标签是collection,这里我们还是使用延迟加载方式,实现原理就是执行两次SQL,按需执行SQL语句。
<mapper namespace="com.tianjunwei.lazy.entity.Teacher">
<select id="getById" parameterType="int" resultMap="user">
select * from users where users.teacher_id=#{id}
</select>
<select id="getTeacher" resultMap="teacher">
select * from tbl_teacher where id=#{id}
</select>
<resultMap type="com.tianjunwei.lazy.entity.Teacher" id="teacher" >
<id column="id" property="id" javaType="int" jdbcType="INTEGER"></id>
<result column="name" property="name" javaType="string" jdbcType="VARCHAR"/>
<collection property="users" column="id" select="com.tianjunwei.lazy.entity.Teacher.getById" javaType="list">//使用collection标签
</collection>
</resultMap>
<resultMap type="com.tianjunwei.lazy.entity.User" id="user" >
<id column="id" property="id" javaType="int" jdbcType="INTEGER"></id>
<result column="name" property="names" javaType="string" jdbcType="VARCHAR"/>
<result column="age" property="age" javaType="int" jdbcType="INTEGER"/>
</resultMap>
</mapper>
Main函数如下:
public class LazyMainTeacher {
public static void main(String [] args){
//mybatis的配置文件
String resource = "learn/mybatis-config.xml";
InputStream is = LazyMainTeacher.class.getClassLoader().getResourceAsStream(resource);
//构建sqlSession的工厂
SqlSessionFactory sessionFactory = new SqlSessionFactoryBuilder().build(is);
SqlSession session = sessionFactory.openSession();
String statement = "com.tianjunwei.lazy.entity.Teacher.getTeacher";//映射sql的标识字符串
//执行查询返回一个唯一user对象的sql
Teacher teacher = (Teacher) session.selectList(statement, 1).get(0);
session.commit(true);
System.out.println(teacher.getName());
List<User> users = teacher.getUsers();
System.err.println(users.get(3).getAge());
}
}
运行结果:可以看到确实是执行两次SQL语句
Mybatis源码分析--关联表查询及延迟加载(一)的更多相关文章
- Mybatis源码分析--关联表查询及延迟加载原理(二)
在上一篇博客Mybatis源码分析--关联表查询及延迟加载(一)中我们简单介绍了Mybatis的延迟加载的编程,接下来我们通过分析源码来分析一下Mybatis延迟加载的实现原理. 其实简单来说Myba ...
- MyBatis源码分析(4)—— Cache构建以及应用
@(MyBatis)[Cache] MyBatis源码分析--Cache构建以及应用 SqlSession使用缓存流程 如果开启了二级缓存,而Executor会使用CachingExecutor来装饰 ...
- MyBatis 源码分析系列文章导读
1.本文速览 本篇文章是我为接下来的 MyBatis 源码分析系列文章写的一个导读文章.本篇文章从 MyBatis 是什么(what),为什么要使用(why),以及如何使用(how)等三个角度进行了说 ...
- Mybatis源码分析之Cache二级缓存原理 (五)
一:Cache类的介绍 讲解缓存之前我们需要先了解一下Cache接口以及实现MyBatis定义了一个org.apache.ibatis.cache.Cache接口作为其Cache提供者的SPI(Ser ...
- MyBatis源码分析-SQL语句执行的完整流程
MyBatis 是支持定制化 SQL.存储过程以及高级映射的优秀的持久层框架.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyBatis 可以对配置和原生Map使用简 ...
- MyBatis源码分析之环境准备篇
前言 之前一段时间写了[Spring源码分析]系列的文章,感觉对Spring的原理及使用各方面都掌握了不少,趁热打铁,开始下一个系列的文章[MyBatis源码分析],在[MyBatis源码分析]文章的 ...
- Mybatis源码分析-BaseExecutor
根据前文Mybatis源码分析-SqlSessionTemplate的简单分析,对于SqlSession的CURD操作都需要经过Executor接口的update/query方法,本文将分析下Base ...
- 【MyBatis源码分析】环境准备
前言 之前一段时间写了[Spring源码分析]系列的文章,感觉对Spring的原理及使用各方面都掌握了不少,趁热打铁,开始下一个系列的文章[MyBatis源码分析],在[MyBatis源码分析]文章的 ...
- MyBatis 源码分析 - 缓存原理
1.简介 在 Web 应用中,缓存是必不可少的组件.通常我们都会用 Redis 或 memcached 等缓存中间件,拦截大量奔向数据库的请求,减轻数据库压力.作为一个重要的组件,MyBatis 自然 ...
随机推荐
- flask 操作mysql的两种方式-sqlalchemy操作
flask 操作mysql的两种方式-sqlalchemy操作 二.ORM sqlalchemy操作 #coding=utf-8 # model.py from app import db class ...
- 洛谷 P2590 [ZJOI2008]树的统计(树链剖分)
题目描述一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w. 我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. QMAX u v ...
- java8接口定义增强
java1.7之前,接口中只允许有全局常量和抽象方法,而1.8之后允许在接口中扩充default修饰的普通方法和static修饰的静态方法 其目的是在修改接口中方法的时候,子类就不必去一一修改 pac ...
- [BZOJ]2589: Spoj 10707 Count on a tree II
Time Limit: 20 Sec Memory Limit: 400 MB Description 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v),你需要回答u xor last ...
- 习题9-4 uva 1630
题意: 给你一串数字,要求你对其进行折叠使其长度最短. 折叠情况:全是一个字母 & 重复的字符串 AAAAAAAAAABABABCCD --> 9(A)3(AB)CCD NEE ...
- bzoj2560串珠子 状压dp+容斥(?)
2560: 串珠子 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 515 Solved: 348[Submit][Status][Discuss] ...
- python常用命令(持续) | Commonly used Python command list (con't)
---------------------------------------------------------------------------------------------------- ...
- Zend引擎探索 之 PHP中前置递增不返回左值
首先来讲,一般我们对"左值"的理解就是可以出现在赋值运算符的左侧的标识符,也就是可以被赋值.这样讲也许并不十分确切,在不同的语言中对左值的定义也不尽相同.在这里我们讨论前置递增(和 ...
- 谈谈Python中的decorator装饰器,如何更优雅的重用代码
众所周知,Python本身有很多优雅的语法,让你能用一行代码写出其他语言很多行代码才能做的事情,比如: 最常用的迭代(eg: for i in range(1,10)), 列表生成式(eg: [ x* ...
- mybatis choose标签的使用
MyBatis 提供了 choose 元素.if标签是与(and)的关系,而 choose 是或(or)的关系. choose标签是按顺序判断其内部when标签中的test条件出否成立,如果有一个成立 ...