缓存的作用
首先缓存的合理使用是优化中最常见的,将从数据库中查询出来的数据放入缓存中,下次使用时不必从数据库查询,而是直接从缓存中读取,避免频繁操作数据库,减轻数据的压力,同时提高系统性能。
为什么需要缓存
BS架构里面,用户的所有操作都是对数据库的增删改查,其中查询的操作是最多的,但如果用户想要某个数据时每次都去数据库查询,这无疑会增加数据库的压力,而且获取时间效率也会降低,所以为了解决这些问题,缓存应用而生,使用了缓存之后,服务器只需要查询一次数据库,然后将数据保存到服务器主机的内存中,以后读取时就直接取内存中的数据,而不需要每次都查数据库,这种方案除了降低数据库压力之外,还提高了响应速度,简直一箭双雕哇~
一级缓存
一级缓存是SqlSession级别的缓存,在操作数据库时需要构造SqlSession对象,在对象中有一个数据结构用于存储缓存数据。不同的sqlSession之间的缓存数据区域是互相不影响的。也就是他只能作用在同一个sqlSession中,不同的sqlSession中的缓存是互相不能读取的。
一级缓存的工作原理:
一级缓存默认是开启的,它在一个sqlSession会话里面的所有查询操作都会保存到缓存中,一般来说一个请求中的所有增删改查操作都是在同一个sqlSession里面的,所以我们可以认为每个请求都有自己的一级缓存,如果同一个sqlSession会话中2 个查询中间有一个 insert 、update或delete 语句,那么之前查询的所有缓存都会清空;
1、用户发起查询请求,查询某条数据,sqlSession先去缓存中查找,是否有该条数据,如果有,则读取并返回;如果没有,则需要从数据库中查询,并将查询到的数据放入到一级缓存区域,供下次查找使用。
2、sqlSession执行增删改操作并执行commit操作,则会清空缓存,这样做的目的是避免脏读。
3、如果commit不清空缓存,会有以下场景:A查询了某商品库存为10件,并将10件库存的数据存入缓存中,之后被客户买走了10件,数据被delete了,但是下次查询这件商品时,并不从数据库中查询,而是从缓存中查询,就会出现错误。
4、示例代码
public static void main(String[] args) throws IOException {
// 加载mybatis配置文件
Reader reader = Resources.getResourceAsReader("config/configuration.xml");
//创建数据工厂
SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();
SqlSessionFactory sqlSessionFactory = builder.build(reader);
SqlSession sqlSession = sqlSessionFactory.openSession(true);
// 获取mapper接口对象
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
// 查询第一次
User user = mapper.selectByPrimaryKey("3rfrf34r34");
// 第二次查询
User user1 = mapper.selectByPrimaryKey("3rfrf34r34");
System.out.println("两个user对象是否相等:"+(user == user1));
//释放会话
sqlSession.clearCache();
sqlSession.close();
}
打印结果:
根据结果可以看到,代码中执行了2次查询, 但实际运行时只查询了一次数据库,第二次获取数据时直接从缓存中读取,并且2次读取的数据都是一样的,到这里,一级缓存就已经生效了;
接下来我们来测试第二种情况 :查询 -> 修改 -> 查询
示例代码:
public static void main(String[] args) throws IOException {
// 加载mybatis配置文件
Reader reader = Resources.getResourceAsReader("config/configuration.xml");
//创建数据工厂
SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();
SqlSessionFactory sqlSessionFactory = builder.build(reader);
SqlSession sqlSession = sqlSessionFactory.openSession(true);
// 获取mapper接口对象
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
// 查询第一次
User user = mapper.selectByPrimaryKey("3rfrf34r34");
// 修改
mapper.updateByPrimaryKey(user);
// 第二次查询
User user1 = mapper.selectByPrimaryKey("3rfrf34r34");
System.out.println("两个user对象是否相等:"+(user == user1));
//释放会话
sqlSession.clearCache();
sqlSession.close();
}
打印结果:
控制台打印了三次sql,其中第一个查询和第三个查询是一样的,但是并没有使用缓存**,是因为每次增删改操作都有可能会改变原来的数据,所以必须刷新缓存**。
二级缓存
二级缓存是mapper级别的缓存,多个SqlSession去操作同一个Mapper的sql语句,多个SqlSession可以共用二级缓存,二级缓存是跨SqlSession的。二级缓存的作用范围更大。
二级缓存流程图:
第一种配置方式
单个mapper配置,主需要在需要开启二级缓存的mapper.xml文件中加入以下配置即可开启。
<!-- 开启单个mapper的二级缓存,也叫全局缓存-->
<cache />
注意一定要加到xxMapper.xml的文件内,千万不要加到mybatis 的主配置文件里面了,会报错的。
第二种配置方式
所有的mapper都开启二级缓存,在mybatis.xml主配置文件中加入以下配置即可
<settings>
<!-- 开启所有mapper的二级缓存 -->
<!--<setting name="cacheEnabled" value="true" />-->
</settings>
示例代码:
public static void main(String[] args) throws IOException {
// 加载mybatis配置文件
Reader reader = Resources.getResourceAsReader("config/configuration.xml");
//创建数据工厂
SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();
SqlSessionFactory sqlSessionFactory = builder.build(reader);
// 第一个会话
SqlSession sqlSession = sqlSessionFactory.openSession(true);
// 获取会话一的mapper接口对象
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
// 第一次查询
User user = mapper.selectByPrimaryKey("3rfrf34r34");
//释放第一个会话
sqlSession.clearCache();
sqlSession.close();
// 第二个会话
SqlSession sqlSession2 = sqlSessionFactory.openSession(true);
// 获取会话二的mapper接口对象
UserMapper mapper2 = sqlSession2.getMapper(UserMapper.class);
// 第二次查询
User user1 = mapper2.selectByPrimaryKey("3rfrf34r34");
// 释放第二个会话
sqlSession2.clearCache();
sqlSession2.close();
}
打印结果:
打印结果很明显,2次查询,但是日志显示只查询了一次数据库, 第二次是从缓存中获取的数据,至此,二级缓存已开启!