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源码分析--关联表查询及延迟加载(一)的更多相关文章

  1. Mybatis源码分析--关联表查询及延迟加载原理(二)

    在上一篇博客Mybatis源码分析--关联表查询及延迟加载(一)中我们简单介绍了Mybatis的延迟加载的编程,接下来我们通过分析源码来分析一下Mybatis延迟加载的实现原理. 其实简单来说Myba ...

  2. MyBatis源码分析(4)—— Cache构建以及应用

    @(MyBatis)[Cache] MyBatis源码分析--Cache构建以及应用 SqlSession使用缓存流程 如果开启了二级缓存,而Executor会使用CachingExecutor来装饰 ...

  3. MyBatis 源码分析系列文章导读

    1.本文速览 本篇文章是我为接下来的 MyBatis 源码分析系列文章写的一个导读文章.本篇文章从 MyBatis 是什么(what),为什么要使用(why),以及如何使用(how)等三个角度进行了说 ...

  4. Mybatis源码分析之Cache二级缓存原理 (五)

    一:Cache类的介绍 讲解缓存之前我们需要先了解一下Cache接口以及实现MyBatis定义了一个org.apache.ibatis.cache.Cache接口作为其Cache提供者的SPI(Ser ...

  5. MyBatis源码分析-SQL语句执行的完整流程

    MyBatis 是支持定制化 SQL.存储过程以及高级映射的优秀的持久层框架.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyBatis 可以对配置和原生Map使用简 ...

  6. MyBatis源码分析之环境准备篇

    前言 之前一段时间写了[Spring源码分析]系列的文章,感觉对Spring的原理及使用各方面都掌握了不少,趁热打铁,开始下一个系列的文章[MyBatis源码分析],在[MyBatis源码分析]文章的 ...

  7. Mybatis源码分析-BaseExecutor

    根据前文Mybatis源码分析-SqlSessionTemplate的简单分析,对于SqlSession的CURD操作都需要经过Executor接口的update/query方法,本文将分析下Base ...

  8. 【MyBatis源码分析】环境准备

    前言 之前一段时间写了[Spring源码分析]系列的文章,感觉对Spring的原理及使用各方面都掌握了不少,趁热打铁,开始下一个系列的文章[MyBatis源码分析],在[MyBatis源码分析]文章的 ...

  9. MyBatis 源码分析 - 缓存原理

    1.简介 在 Web 应用中,缓存是必不可少的组件.通常我们都会用 Redis 或 memcached 等缓存中间件,拦截大量奔向数据库的请求,减轻数据库压力.作为一个重要的组件,MyBatis 自然 ...

随机推荐

  1. [转]scrapy中的logging

    logging模块是Python提供的自己的程序日志记录模块. 在大型软件使用过程中,出现的错误有时候很难进行重现,因此需要通过分析日志来确认错误位置,这也是写程序时要使用日志的最重要的原因. scr ...

  2. Http读书笔记1-5章

    第一章 内容提要 这一章主要介绍了什么是http以及http是干嘛的,以及与之有关的相关概念,当然了这些概念都是概览式的介绍一些.所以我将采用问答式的方式描述这一章! Q:http是干嘛的? A:ht ...

  3. 洛谷 P3038 [USACO11DEC]牧草种植Grass Planting(树链剖分)

    题解:仍然是无脑树剖,要注意一下边权,然而这种没有初始边权的题目其实和点权也没什么区别了 代码如下: #include<cstdio> #include<vector> #in ...

  4. C# 删除文件夹

    三种方法 1.这种方法简单,能删除文件夹内的所有文件(文件及子目录) DirectoryInfo di = new DirectoryInfo(string Path);         di.Del ...

  5. AtCoder Grand Contest 021 D - Reversed LCS

    Description Takahashi has decided to give a string to his mother. The value of a string T is the len ...

  6. linux x86内核中的分页机制

    Linux采用了通用的四级分页机制,所谓通用就是指Linux使用这种分页机制管理所有架构的分页模型,即便某些架构并不支持四级分页.对于常见的x86架构,如果系统是32位,二级分页模型就可满足系统需求: ...

  7. bzoj3561DZY Loves Math VI

    3561: DZY Loves Math VI Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 503  Solved: 333[Submit][Sta ...

  8. ubuntu linux c学习笔记----共享内存(shmget,shmat,shmdt,shmctl)

    shmget int shmget(key_t key, size_t size, int flag); key: 标识符的规则 size:共享存储段的字节数 flag:读写的权限 返回值:成功返回共 ...

  9. OCP 认证考试报名费技巧题库051052053解析合格线

    本人于2017年4月22日通过参加OCP考试,第一次参加,一天之内考了三门,三门一次性通过,052 - 95% ,053 - 86% ,051 - 100% 一.关于考试考试报名费: 052:158$ ...

  10. solr6.6初探之配置篇

    一.solr的简介 1) solr是企业级应用的全文检索项目,它是基于Apache Lucence搜索引擎开发出来的用于搜索的应用工程 2) solr最新版本6.6 下载地址:下载地址 二 启动与配置 ...