Mybatis的一级、二级缓存
如图所示:
Mybatis的缓存如图所示:
当数据没有改变,开启SQLsession使用SQL语句对数据进行一次查询时,会将数据进行缓存,当第二次查询同样的数据时,则命中缓存,不去查询数据库,加快查询耗时;
特点:
一级缓存、二级缓存都是基于本地缓存的一个名为PerpetualCache,它本质是一个Hashmap
一级缓存作用域是session级别,同一个session,调同一个SQL会进行缓存,默认是打开的
二级缓存作用域更广,是namespace和mapper的作用域,只要是同一个namespace或者查询同一个mapper文件(同一张表)的SQL,都会进行缓存,不依赖session,但是默认是关闭的;
一、一级缓存
1、基于一个session
基于perpetualCache的HashMap本地缓存,其作用域为Session,当Session进行flush或close之后,该Session中的所有Cache就将清空,默认打开一级缓存
// 获取一个sqlsession,用它来执行SQL
Sqlsession sqlsession = sqlSessionFactory.openSession();
// 执行SQL,用同一个session对象获取两次,同一笔数据,
// 得到userMapper1、userMapper2两个代理对象
UserMapper userMapper1 = sqlsession.getMapper(UserMapper.class);
UserMapper userMapper1 = sqlsession.getMapper(UserMapper.class);
// 用这两个代理对象,执行两次查询SQL,但是到数据库层面只会进行一次查询
User user = userMapper1.selectById(6);
System.out.prientln(user);
System.out.prientln("--------------------------");
User user1 = userMapper2.selectById(6);
System.out.prientln(user1);
控制台结果如下:
我们可以看到,明明调用了两次select语句,但是实际上只查询了一次数据库,证明后面一次直接走的是缓存,这就是一级缓存
因为他们用的是同一个session,走了一级缓存,用两个session就会查询两次数据库了
2、基于两个session
// 获取一个sqlsession,用它来执行SQL
Sqlsession sqlsession1 = sqlSessionFactory.openSession();
// 执行SQL,用同一个session对象获取两次,同一笔数据,
// 得到userMapper1代理对象
UserMapper userMapper1 = sqlsession1.getMapper(UserMapper.class);
// 用这两个代理对象,执行两次查询SQL,但是到数据库层面只会进行一次查询
User user = userMapper1.selectById(6);
System.out.prientln(user);
sqlsession1.close();
// 另外开启一个session
Sqlsession sqlsession2 = sqlSessionFactory.openSession();
System.out.prientln("--------------------------");
UserMapper userMapper2 = sqlsession2.getMapper(UserMapper.class);
User user2 = userMapper2.selectById(6);
System.out.prientln(user2);
sqlsession2.close();
控制台结果如下:
可以看到,以及缓存失效了,查询了两次数据库;
另外:当数据有过增删改等操作后,缓存就会失效,也会查两次数据库
那如果我两个session,也想使用Mybatis的缓存,可以不可以呢?
可以的,那就需要使用二级缓存了:
二、二级缓存
二级缓存也是基于perpetualCache的HashMap本地缓存,其作用域是namespace和mapper;而不是依赖于SQL session,所以范围更广
当两个session不能缓存时,我们可以手动配置开启二级缓存,使不同的session也能缓存:
开启方式,两步:
1、全局配置文件:
<settings>
<setting name='cacheEnabled" value='true'
</settings>
2、mapper映射文件中:
使用<cache/>标签让当前mapper文件生效二级缓存
我们测试一下,还是上面那个一级缓存失效的例子:
// 获取一个sqlsession,用它来执行SQL
Sqlsession sqlsession1 = sqlSessionFactory.openSession();
// 执行SQL,用同一个session对象获取两次,同一笔数据,
// 得到userMapper1代理对象
UserMapper userMapper1 = sqlsession1.getMapper(UserMapper.class);
// 用这两个代理对象,执行两次查询SQL,但是到数据库层面只会进行一次查询
User user = userMapper1.selectById(6);
System.out.prientln(user);
sqlsession1.close();
// 另外开启一个session
Sqlsession sqlsession2 = sqlSessionFactory.openSession();
System.out.prientln("--------------------------");
UserMapper userMapper2 = sqlsession2.getMapper(UserMapper.class);
User user2 = userMapper2.selectById(6);
System.out.prientln(user2);
sqlsession2.close();
控制台结果如下:
手动二级缓存后,缓存生效
注意事项:
1、对于缓存数据更新机制,当某一个作用域(一级缓存Session/二级缓存namespace)的数据,进行了新增、修改、删除操作后,默认该作用域下的所有select中的缓存将被清除clear
2、二级缓存需要缓存中的数据,实现Serializable接口
3、只有会话提交或者关闭后,一级缓存中的数据才会转移到二级缓存中