Mybatis的一级缓存和二级缓存详解

一、缓存是什么

缓存其实就是存储在内存中的临时数据。这里的数据量会比较小,一般来说,服务器的内存也是有限的,不可能将所有的数据都放到服务器的内存里面,所以, 只会把关键数据放到缓存中,缓存因为速度快,使用方便而出名!

二、为什么需要缓存

BS架构里面,用户的所有操作都是对数据库的增删改查,其中查询的操作是最多的,但如果用户想要某个数据时每次都去数据库查询,这无疑会增加数据库的压力,而且获取时间效率也会降低,所以为了解决这些问题,缓存应用而生,使用了缓存之后,服务器只需要查询一次数据库,然后将数据保存到服务器主机的内存中,以后读取时就直接取内存中的数据,而不需要每次都查数据库,这种方案除了降低数据库压力之外,还提高了响应速度。

三、哪些数据会放到缓存

通常情况下,都会将那些变化较少且经常用到的数据会放到缓存中,比如像字典、系统参数、有固定值的状态码等等;另外将用户保存到缓存也是一种很好的策略,这样登录的时候就可以极速响应了。

四、mybatis一级缓存

一级缓存是默认开启的,它在一个sqlSession会话里面的所有查询操作都会保存到缓存中,一般来说一个请求中的所有增删改查操作都是在同一个sqlSession里面的,所以我们可以认为每个请求都有自己的一级缓存,如果同一个sqlSession会话中2 个查询中间有一个 insert 、update或delete 语句,那么之前查询的所有一级缓存(执行更新操作的效果)、二级缓存(执行commit()的效果)都会清空。一级缓存没有过期时间,不会自动清空,只有在执行更新操作(update、delete、insert)后并执行commit时或关闭sqlSession会话时清空缓存。

 示例代码一:

复制代码

 1 public static void main(String[] args) throws IOException {
 2  
 3         // 加载mybatis配置文件
 4         Reader reader = Resources.getResourceAsReader("config/configuration.xml");
 5         //创建数据工厂
 6         SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();
 7         SqlSessionFactory sqlSessionFactory = builder.build(reader);
 8         SqlSession sqlSession = sqlSessionFactory.openSession(true);
 9  
10  
11         // 获取mapper接口对象
12         UserMapper mapper = sqlSession.getMapper(UserMapper.class);
13  
14         // 查询第一次
15         User user = mapper.selectByPrimaryKey("3rfrf34r34");
16         // 第二次查询
17         User user1 = mapper.selectByPrimaryKey("3rfrf34r34");
18         System.out.println("两个user对象是否相等:"+(user == user1));
19         //释放会话
20         sqlSession.clearCache();
21         sqlSession.close();
22     }

复制代码

打印结果:(根据结果可以看到,代码中执行了2次查询, 但实际运行时只查询了一次数据库,第二次获取数据时直接从缓存中读取,并且2次读取的数据都是一样的,到这里,一级缓存就已经生效了;)

示例代码二:查询 -> 修改 -> 查询

复制代码

 1 public static void main(String[] args) throws IOException {
 2  
 3         // 加载mybatis配置文件
 4         Reader reader = Resources.getResourceAsReader("config/configuration.xml");
 5         //创建数据工厂
 6         SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();
 7         SqlSessionFactory sqlSessionFactory = builder.build(reader);
 8         SqlSession sqlSession = sqlSessionFactory.openSession(true);
 9  
10  
11         // 获取mapper接口对象
12         UserMapper mapper = sqlSession.getMapper(UserMapper.class);
13         // 查询第一次
14         User user = mapper.selectByPrimaryKey("3rfrf34r34");
15         
16         // 修改
17         mapper.updateByPrimaryKey(user);
18         // 第二次查询
19         User user1 = mapper.selectByPrimaryKey("3rfrf34r34");
20         System.out.println("两个user对象是否相等:"+(user == user1));
21         //释放会话
22         sqlSession.clearCache();
23         sqlSession.close();
24     }

复制代码

打印结果:(控制台打印了三次sql,其中第一个查询和第三个查询是一样的,但是并没有使用缓存,为什么会这样呢?因为每次增删改操作都有可能会改变原来的数据,所以必须刷新缓存;)

 五、mybatis二级缓存

二级缓存是全局的,指的是Mybatis中SqlSessionFactory对象的缓存,也就是说;多个请求可以共用一个缓存,二级缓存需要手动开启,有2种方式配置二级缓存,缓存会先放在一级缓存中,当sqlSession会话提交或者关闭时才会将一级缓存刷新到二级缓存中;开启二级缓存后,用户查询时,会先去二级缓存中找,找不到在去一级缓存中找。二级缓存有过期时间,每当存取数据的时候,都有检测一下cache的生命时间,默认是1小时,如果这个cache存活了一个小时,那么将整个清空一下。是 Mapper 级别的缓存,多个 SqlSession 去操作同一个 Mapper 的 sql 语句,多个 SqlSession 可以共用二级缓存,二级缓存是跨 SqlSession 的。

1、第一种配置方式:

单个mapper配置,主需要在需要开启二级缓存的mapper.xml文件中加入以下配置即可开启

 2、第二种配置方式:

所有的mapper都开启二级缓存,在mybatis.xml中加入以下配置即可

 示例代码:

复制代码

 1 public static void main(String[] args) throws IOException {
 2  
 3         // 加载mybatis配置文件
 4         Reader reader = Resources.getResourceAsReader("config/configuration.xml");
 5         //创建数据工厂
 6         SqlSessionFactoryBuilder builder = new SqlSessionFactoryBuilder();
 7  
 8         SqlSessionFactory sqlSessionFactory = builder.build(reader);
 9  
10         // 第一个会话
11         SqlSession sqlSession = sqlSessionFactory.openSession(true);
12  
13  
14         // 获取会话一的mapper接口对象
15         UserMapper mapper = sqlSession.getMapper(UserMapper.class);
16  
17         // 第一次查询
18         User user = mapper.selectByPrimaryKey("3rfrf34r34");
19         
20         //释放第一个会话
21         sqlSession.clearCache();
22         sqlSession.close();
23         // 第二个会话
24         SqlSession sqlSession2 = sqlSessionFactory.openSession(true);
25         // 获取会话二的mapper接口对象
26         UserMapper mapper2 = sqlSession2.getMapper(UserMapper.class);
27         // 第二次查询
28         User user1 = mapper2.selectByPrimaryKey("3rfrf34r34");
29         // 释放第二个会话
30         sqlSession2.clearCache();
31         sqlSession2.close();
32     }

复制代码

打印结果:(2次查询,但是日志显示只查询了一次数据库, 第二次是从缓存中获取的数据,至此,二级缓存已开启!)

缓存会被视为读/写缓存,这意味着获取到的对象并不是共享的,可以安全地被调用者修改,而不干扰其他调用者或线程所做的潜在修改。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值