MyBatis缓存技术(一级缓存、二级缓存)

1、MyBatis的缓存分类

一级缓存

一级缓存的作用域是一个SqlSession。MyBatis默认开启一级缓存。在同一个SqlSession中,执行相同的查询SQL,第一次会查询数据库,并写到缓存中;第二次直接从缓存中读取。当执行SQL时,两次查询中间发生了增、删、改的操作,则SqlSession的缓存会被清空。

二级缓存

二级缓存的作用域是一个namesp下的mapper映射文件内容,可以多个SqlSession共享。MyBatis需要手动设置启动二级缓存(在SQLConfig中配置setting标签)。在同一个namespace下的mapper文件中,执行相同的SQL,第一次会去查询数据库,并写到缓存中;第二次直接从缓存中取。当执行SQL时,两次查询中间发生了增、删、改的操作,则二级缓存清空。

2、MyBatis缓存原理

2.1、一级缓存


一级缓存是根据SqlSession为单位划分的。每次查询会先去缓存中找,找不到再去数据库中查询,然后把结果写到缓存中。MyBatis的内部缓存使用一个HashMap,key为hashcode+statementId+sql语句。value为查询出来的结果映射成的Java对象。

2.2、二级缓存原理


二级缓存是mapper级别的。MyBatis默认是不开启二级缓存的。

第一次调用mapper下的SQL去查询用户信息。查询到的信息会存到该mapper对应的二级缓存区域内;第二次调用相同namespace下的mapper映射文件中相同的SQL去查询用户信息,MyBatis会自动去对应的二级缓存中取结果。如果相同namespace下的mapper映射文件中执行了增、删、改的SQL,并且执行了commi操作,此时会清空该namespace下的二级缓存。

3、如何开启二级缓存

3.1、配置SqlConfig.xml,开启二级缓存总开关

<!-- 全局配置参数 -->
<settings>
	<!-- 开启二级缓存 -->
	<setting name="cacheEnabled" value="true"/>
</settings>
3.2、在mapper.xml中开启二级缓存

<!-- 开启该mapper的namespace下的二级缓存 -->
<cache/>
cache标签的参数说明:

flushInterval                     刷新间隔,可以被设置为任意正整数,单位为毫秒。默认情况是不设置的,也就是没有刷新间隔,缓存仅仅调用增、删、改语句时刷新
size引用数目,可以被设置成任意正整数,缓存的内存容量,默认是1024
readOnly只读,可以被设置为true或false,默认是false。只读的缓存会给所有调用者返回缓存对象的相同实例,这提供了很重要的性能优势。可读写的缓存会返回缓存对象的拷贝对象(通过序列化),这会慢一些,但是安全。

例如:

<cache flushInterval="60000" size="512" readOnly="true"/>
3.3、让POJO对象实现序列化接口

package org.mybatis.demo.po;

import java.io.Serializable;
import java.util.Date;

public class User implements Serializable {

	private static final long serialVersionUID = 1L;
	// 属性名和数据库表的字段对应
	private int id;
	private String username;// 用户姓名
	private String sex;// 性别
	private Date birthday;// 生日
	private String address;// 地址

	public int getId() {
		return id;
	}

	public void setId(int id) {
		this.id = id;
	}

	public String getUsername() {
		return username;
	}

	public void setUsername(String username) {
		this.username = username;
	}

	public String getSex() {
		return sex;
	}

	public void setSex(String sex) {
		this.sex = sex;
	}

	public Date getBirthday() {
		return birthday;
	}

	public void setBirthday(Date birthday) {
		this.birthday = birthday;
	}

	public String getAddress() {
		return address;
	}

	public void setAddress(String address) {
		this.address = address;
	}

	@Override
	public String toString() {
		return "User [id=" + id + ", username=" + username + ", sex=" + sex
				+ ", birthday=" + birthday + ", address=" + address + "]";
	}
}
注意:如果该类存在父类,那么父类也要实现序列化

4、禁用二级缓存

在statement中的select标签设置userCache属性值为false可以禁用当前select语句的二级缓存,即每次查询都是去数据库中查询,默认值是true

<select id="findUserById" parameterType="java.lang.Integer" resultType="org.mybatis.demo.po.User" useCache="false">
	select * from user where id = #{id}
</select>
5、刷新二级缓存

在statement中的select标签设置flushCache属性值为false可以设置不执行缓存刷新,即每次执行增、删、改操作时,缓存不清空

<select id="findUsers" resultType="org.mybatis.demo.po.User" flushCache="false">
	select * from user
</select>
注意:如果不执行刷新缓存,会出现脏读。

6、源代码

MyBatis缓存(一级缓存、二级缓存)





  • 5
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值