MyBatis缓存

MyBatis缓存

使⽤缓存可以减少 Java 应⽤与数据库的交互次数,从⽽提升程序的运⾏效率。⽐如查询出 id = 1 的对
象,第⼀次查询出之后会⾃动将该对象保存到缓存中,当下⼀次查询时,直接从缓存中取出对象即可,
⽆需再次访问数据库

缓存的分类:

⼀级缓存:SqlSession 级别,默认开启,并且不能关闭。

操作数据库时需要创建 SqlSession 对象,在对象中有⼀个 HashMap ⽤于存储缓存数据,不同的
SqlSession 之间缓存数据区域是互不影响的。

⼀级缓存的作⽤域是 SqlSession 范围的,当在同⼀个 SqlSession 中执⾏两次相同的 SQL 语句事,第⼀
次执⾏完毕会将结果保存到缓存中,第⼆次查询时直接从缓存中获取。

需要注意的是,如果 SqlSession 执⾏了 DML 操作(insert、update、delete),MyBatis 必须将缓存
清空以保证数据的准确性。

public class Test {
	public static void main(String[] args) {
		InputStream inputStream = Resource.getResourceAsStream("config.xml");
		SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();
		SqlSessionFactory sqlSessionFactory = sqlSessionFactoryBuilder.build(inputStream);
		SqlSession sqlSession = sqlSessionFactory.openSession();
		AccountRepository accountRepository = sqlSession.getMapper(AccountRepository.class);
		Account account = accountRepository.findById(1L);
		System.out.println(account);
		Account account1 = accountRepository.findById(1L);
		System.out.println(account1);
		sqlSession.close();
 	}
}

通过SQL打印的日志可以发现,虽然查询了两次,但是只调用了一次SQL。

⼆级缓存:Mapper 级别,默认关闭,可以开启。

使⽤⼆级缓存时,多个 SqlSession 使⽤同⼀个 Mapper 的 SQL 语句操作数据库,得到的数据会存在⼆
级缓存区,同样是使⽤ HashMap 进⾏数据存储,相⽐较于⼀级缓存,⼆级缓存的范围更⼤,多个
SqlSession 可以共⽤⼆级缓存,⼆级缓存是跨 SqlSession 的。

⼆级缓存是多个 SqlSession 共享的,其作⽤域是 Mapper 的同⼀个 namespace,不同的 SqlSession
两次执⾏相同的 namespace 下的 SQL 语句,参数也相等,则第⼀次执⾏成功之后会将数据保存到⼆级
缓存中,第⼆次可直接从⼆级缓存中取出数据。

1、MyBatis自带的二级缓存

  • 在mybatis全局配置文件中开启二级缓存
<setting name="cacheEnable" value="true"/>
  • 在Mapper中配置二级缓存
<!--只需要添加cache空标签即可-->
<cache></cache>
  • 实体类实现序列化接口
@Data
@AllArgsConstructor
@NoArgsConstructor
public class Account implements Serializable {
 	private long id;
 	private String username;
 	private String password;
	private int age;
}
public class Test {
	public static void main(String[] args) {
		InputStream inputStream = Resource.getResourceAsStream("config.xml");
		SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();
		SqlSessionFactory sqlSessionFactory = sqlSessionFactoryBuilder.build(inputStream);
		SqlSession sqlSession = sqlSessionFactory.openSession();
		AccountRepository accountRepository = sqlSession.getMapper(AccountRepository.class);
		Account account = accountRepository.findById(1L);
		System.out.println(account);
		sqlSession.close();
		SqlSession sqlSession = sqlSessionFactory.openSession();
		AccountRepository accountRepository = sqlSession.getMapper(AccountRepository.class);
		Account account = accountRepository.findById(1L);
		System.out.println(account1);
		
 	}
}

此时关闭了第一个sqlsession,重新创建了一个新sqlsession,但是还是只执行了一条sql语句

2、ehcache二级缓存

  • 添加相关依赖
<dependency>
	<groupId>org.mybatis</groupId>
 	<artifactId>mybatis-ehcache</artifactId>
 	<version>1.0.0</version>
</dependency>
<dependency>
 	<groupId>net.sf.ehcache</groupId>
 	<artifactId>ehcache-core</artifactId>
 	<version>2.4.3</version>
</dependency>
  • 添加ehcache.xml
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 xsi:noNamespaceSchemaLocation="../config/ehcache.xsd">
	<diskStore/>
 	<defaultCache
 		maxElementsInMemory="1000"
 		maxElementsOnDisk="10000000"
 		eternal="false"
 		overflowToDisk="false"
 		timeToIdleSeconds="120"
 		timeToLiveSeconds="120"
 		diskExpiryThreadIntervalSeconds="120"
 		memoryStoreEvictionPolicy="LRU">
 	</defaultCache>
</ehcache>
  • MyBatis配置文件中开启二级缓存
<setting name="cacheEnable" value="true"/>
  • Mapper中配置二级缓存
<cache type="org.mybatis.caches.ehcache.EhcacheCache">
 	<!-- 缓存创建之后,最后⼀次访问缓存的时间⾄缓存失效的时间间隔 -->
 	<property name="timeToIdleSeconds" value="3600"/>
 	<!-- 缓存⾃创建时间起⾄失效的时间间隔 -->
 	<property name="timeToLiveSeconds" value="3600"/>
 	<!-- 缓存回收策略,LRU表示移除近期使⽤最少的对象 -->
 	<property name="memoryStoreEvictionPolicy" value="LRU"/>
</cache>
  • 实体类不需要实现序列化接口
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值