延迟加载也称为“按需加载”。在执行关联查询时候,mybatis不会把对象的关联实体查询出来,只有在实际使用到关联实体的时候,才会把关联实体查询出来。因此,延迟加载可以减少不必要的关联查询,提高数据库的查询效率。
一般来说,延迟加载在一对多或多对多关系中使用较多。
配置延迟加载
首先,把mybatis安装目录下的asm-7.0.jar和cglib-3.2.10.jar文件拷贝到项目中;
接着,按照下面方式配置延迟加载。
方式一:局部延迟加载。
局部延迟策略就是只对某个实体类启用延迟加载策略
指定fetchType="lazy"属性即可。
<resultMap id="categoryResultMap" type="categoryAlias">
<id property="id" column="id" />
<id property="name" column="name" />
<!-- 配置一对多的关系
property:集合的属性名
ofType:集合元素的类型
column:当前实体对应表的主键名
select:指定关联查询的SQL语句
fetchType: 指定是否延迟加载,lazy代表延迟加载,eager代表不延迟加载 -->
<collection property="products" ofType="productAlias" column="id"
select="mybatisqs.demo01helloword.ProductMapper.selectProductByCategoryId"
fetchType="lazy"/>
</resultMap>
如果是使用注解配置方式,可以在@Many注解中指定fetchType=FetchType.LAZY属性即可。例如:
@Select("select * from category where id = #{categoryId}")
@Results({
@Result(property="id", column="id"),
@Result(property="name", column="name"),
@Result(property="products", column="id", many=@Many(
select="mybatisqs.demo01helloword.ProductMapper.selectProductByCategoryId"
, fetchType=FetchType.LAZY))
})
Category selectCategoryById(int categoryId);
方式二:全局延迟加载
启用全局延迟加载策略后,所有实体类都会启用生效。启用全局延迟策略需要在MyBatis核心配置文件中进行配置
<settings> <setting name="lazyLoadingEnabled" value="true" /> <setting name="aggressiveLazyLoading" value="false"/> </settings>
MyBatis缓存机制
使用缓存的最大好处可以提供数据的查询速度,减轻数据压力,提高数据库性能。
不带缓存的数据库查询:
带缓存的数据库查询
一级缓存(自动打开)
一级缓存也称为Session缓存。同一个Session中的数据会保存在Session中。只要Session不关闭,不管查询N次都会返回相同的结果。
二级缓存
Mybatis里面,二级缓存就是文件级别的缓存。启用二级缓存后,即使Session关闭了,也不会对缓存数据有影响。
MyBatis缓存的执行过程:
如果配置二级缓存,那么执行查询操作时候,mybatis会先从二级缓存中查询数据,如果二级缓存没有,再从一级缓存(session)中查找数据,如果一级缓存也没有,才会查询数据库。
MyBatis整合ehcache
第一步:下载和导入ehcache的jar包;
下载地址:https://github.com/mybatis/ehcache-cache/releases
第二步:在工程src目录下新建一个xml格式文件,该文件的名字为ehcache.xml,然后把下面内容拷贝到文件中;
<?xml version="1.0" encoding="UTF-8"?>
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd">
<!--缓存目录 -->
<diskStore path="d:/ehcache" />
<!--
属性说明:
name: Cache的唯一标识
maxElementsInMemory: 内存中最大缓存对象数
maxElementsOnDisk: 磁盘中最大缓存对象数,若是0表示无穷大
eternal: Element是否永久有效,一但设置了,timeout将不起作用
overflowToDisk: 当内存中Element数量达到maxElementsInMemory时,Ehcache将会Element写到磁盘中
timeToIdleSeconds: 设置Element在失效前允许的闲置时间。仅当element不是永久有效时使用
timeToLiveSeconds: 设置Element在失效前允许存活时间。最大时间介于创建时间和失效时间之间。仅当element不是永久有效时使用
diskExpiryThreadIntervalSeconds: 磁盘失效线程运行时间间隔,默认是120秒
memoryStoreEvictionPolicy: 当达到maxElementsInMemory限制时,Ehcache将会根据指定的策略去清理内存。默认策略是LRU
-->
<defaultCache
name="defaultCache"
maxElementsInMemory="1000"
maxElementsOnDisk="10000000"
eternal="true"
overflowToDisk="true"
timeToIdleSeconds="120"
timeToLiveSeconds="120"
diskExpiryThreadIntervalSeconds="120"
memoryStoreEvictionPolicy="LRU">
</defaultCache>
</ehcache>
、第三步:在需要使用缓存的映射文件中加入cache节点。
<cache type="org.mybatis.caches.ehcache.EhcacheCache"/>
加入了cache后,该映射文件下的所有查询操作都会使用缓存。如果不需要启用缓存,可以在statement中指定useCache="false"属性即可。
MyBatis事务管理机制
MyBatis的事务管理分为两种形式:
1使用JDBC事务管理机制,即使用java.sql.Connection对象完成事物的提交、回滚、关闭等操作;
2使用MANAGED的事务管理机制。在这种机制下,MyBatis自己不会实现事务管理,而是让程序的容器来实现对事务的管理;
如何配置事务管理?
在MyBatis配置文件的enviorment节点中进行配置。例如:
一般来说,如果只是单独使用mybatis来构建本地程序,可以将事务管理设置为JDBC。但是,如果使用mybatis构建WEB应用,我们可以将事务管理设置为MANAGED,即MyBatis自己不管理事务,而是由容器负责管理事务。