【MyBatis框架点滴】——MyBatis二级缓存

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介:   在上篇文章的那张图上稍作修改,就可以很明了的看出来什么是二级缓存,它和一级缓存的区别是什么:


  在上篇文章的那张图上稍作修改,就可以很明了的看出来什么是二级缓存,它和一级缓存的区别是什么:


22.png


 一级缓存是SqlSession级别的缓存,二级缓存则是Mapper级别的缓存,这里的Mapper可以看做是Mapper配置文件。多个SqlSession可以操作同一个Mapper配置文件,所以二级缓存比一级缓存的范围要大,它是跨SqlSession的,多个SqlSession可以共享二级缓存中的数据。


开启二级缓存


 MyBatis默认不开启二级缓存,需要在两个地方手动开启:


 1、在MyBatis全局配置文件(SqlMapConfig.xml)中添加如下配置:


<configuration>
  <settings>
    <!-- 开启二级缓存 -->
    <setting name="cacheEnabled" value="true"/>
  </settings>
</configuration>


  2、在相应Mapper配置文件下添加<cache/>


<mapper namespace="com.danny.mybatis.generator.mapper.UserMapper" >
  <!-- 开启本Mapper下的二级缓存 -->
  <cache/>
</mapper> 


  因为二级缓存的存储介质多种多样,不一定是内存,所以还需要让相应的pojo对象实现序列化接口Serializable。

  这时就可以使用二级缓存了。


二级缓存应用



  实验一

@Test
  public void testCache2() throws Exception {
    SqlSession sqlSession1 = sqlSessionFactory.openSession();
    SqlSession sqlSession2 = sqlSessionFactory.openSession();
    //创建代理对象
    UserMapper userMapper1 = sqlSession1.getMapper(UserMapper.class);
    UserMapper userMapper2 = sqlSession2.getMapper(UserMapper.class);
    // 查询id为1的用户
    User user1 = userMapper1.selectByPrimaryKey(6);
    System.out.println(user1.getUsername());
    sqlSession1.close();
    // 查询id为1的用户
    User user2 = userMapper2.selectByPrimaryKey(6);
    System.out.println(user2.getUsername());    
    sqlSession2.close();      
  }


  日志:


23.png


 由日志可以看出,第一次查询的缓存命中率为0,表示至今为止共执行了1次查询,从缓存中查出数据的次数为0;第二次的缓存命中率为05,表示至今为止共执行了2次查询,从缓存中查出数据的次数为1。


 并且不同的SqlSession查询相同的数据,只有第一次查询执行了sql语句,第二次查询直接从二级缓存中读取。(注意,执行第一次查询后,必须关闭sqlSession,才能将sqlSession中的数据写入到二级缓存区域,如果不关闭,则无法写入二级缓存)



 **实验二**:


@Test
public void testCache2() {
  SqlSession sqlSession1 = sqlSessionFactory.openSession();
  SqlSession sqlSession2 = sqlSessionFactory.openSession();
  SqlSession sqlSession3 = sqlSessionFactory.openSession();
  //创建代理对象
  UserMapper userMapper1 = sqlSession1.getMapper(UserMapper.class);
  UserMapper userMapper2 = sqlSession2.getMapper(UserMapper.class);
  UserMapper userMapper3 = sqlSession3.getMapper(UserMapper.class);
  // 查询id为1的用户
  User user1 = userMapper1.selectByPrimaryKey(6);
  System.out.println(user1.getUsername());
  sqlSession1.close();
  // 更新id为1的用户
  user1.setUsername("DannyHoo");
  userMapper3.updateByPrimaryKey(user1);
  sqlSession3.commit();
  sqlSession3.close();
  // 查询id为1的用户
  User user2 = userMapper2.selectByPrimaryKey(6);
  System.out.println(user2.getUsername());
  sqlSession2.close();
}


  日志:

24.png


  上面日志中看到,第一次查询的缓存命中率为0;由于sqlSession2在第二次查询之前,sqlSession3执行了改变数据库数据的操作,所以会清空该Mapper下二级缓存中的内容,导致第二次查询的缓存命中率也为0。

  两次实验的结果也可以用下面一张图来表示如下:


25.png


扩展



  1、禁用二级缓存

  在Mapper配置文件的<select>标签内添加useCache="false" 属性,可以设置该查询结果不会放到二级缓存中,每次执行该查询都会执行sql语句。


<select id="selectByPrimaryKey" resultMap="BaseResultMap" parameterType="java.lang.Integer" useCache="false">
  select 
    <include refid="Base_Column_List" />
    from user
    where id = #{id,jdbcType=INTEGER}
</select>


 这种情况比较适合对刷新频率较高的数据的查询。



 2、刷新缓存配置

 从上面可以看出,在mapper的同一个namespace中,如果有其它insert、update、delete操作数据后需要刷新缓存,如果不执行刷新缓存会出现脏读。


 设置Mapper中statement的flushCache属性,可以设置缓存是否刷新。默认情况下为true,即执行了增删改之后会刷新二级缓存;为false时则不会刷新。(使用缓存时如果手动修改数据库表中的查询数据会出现脏读哦。)


<update id="updateByPrimaryKeySelective" parameterType="com.danny.mybatis.generator.pojo.User" flushCache="true">
……
</update>


总结:


 一般情况下,每一个Mapper文件都会设置一个namespace,如果两个Mapper文件的namespace一样的话,那么这两个Mapper其实对应同一个二级缓存区域(这两个Mapper执行的sql查询到的数据将存储在同一个二级缓存区域中),由此看来,二级缓存区域是按照Mapper的namespace来分的。


相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
3月前
|
缓存 Java 数据库连接
mybatis复习05,mybatis的缓存机制(一级缓存和二级缓存及第三方缓存)
文章介绍了MyBatis的缓存机制,包括一级缓存和二级缓存的配置和使用,以及如何整合第三方缓存EHCache。详细解释了一级缓存的生命周期、二级缓存的开启条件和配置属性,以及如何通过ehcache.xml配置文件和logback.xml日志配置文件来实现EHCache的整合。
mybatis复习05,mybatis的缓存机制(一级缓存和二级缓存及第三方缓存)
|
8天前
|
缓存 Java 数据库连接
MyBatis缓存机制
MyBatis提供两级缓存机制:一级缓存(Local Cache)默认开启,作用范围为SqlSession,重复查询时直接从缓存读取;二级缓存(Second Level Cache)需手动开启,作用于Mapper级别,支持跨SqlSession共享数据,减少数据库访问,提升性能。
19 1
|
11天前
|
缓存 Java 数据库连接
深入探讨:Spring与MyBatis中的连接池与缓存机制
Spring 与 MyBatis 提供了强大的连接池和缓存机制,通过合理配置和使用这些机制,可以显著提升应用的性能和可扩展性。连接池通过复用数据库连接减少了连接创建和销毁的开销,而 MyBatis 的一级缓存和二级缓存则通过缓存查询结果减少了数据库访问次数。在实际应用中,结合具体的业务需求和系统架构,优化连接池和缓存的配置,是提升系统性能的重要手段。
27 4
|
1月前
|
SQL Java 数据库连接
持久层框架MyBatisPlus
持久层框架MyBatisPlus
42 1
持久层框架MyBatisPlus
|
1月前
|
SQL 缓存 Java
【详细实用のMyBatis教程】获取参数值和结果的各种情况、自定义映射、动态SQL、多级缓存、逆向工程、分页插件
本文详细介绍了MyBatis的各种常见用法MyBatis多级缓存、逆向工程、分页插件 包括获取参数值和结果的各种情况、自定义映射resultMap、动态SQL
【详细实用のMyBatis教程】获取参数值和结果的各种情况、自定义映射、动态SQL、多级缓存、逆向工程、分页插件
|
1月前
|
SQL 缓存 Java
MyBatis如何关闭一级缓存(分注解和xml两种方式)
MyBatis如何关闭一级缓存(分注解和xml两种方式)
67 5
|
3月前
|
缓存 Java 开发工具
Spring是如何解决循环依赖的?从底层源码入手,详细解读Spring框架的三级缓存
三级缓存是Spring框架里,一个经典的技术点,它很好地解决了循环依赖的问题,也是很多面试中会被问到的问题,本文从源码入手,详细剖析Spring三级缓存的来龙去脉。
219 24
Spring是如何解决循环依赖的?从底层源码入手,详细解读Spring框架的三级缓存
|
2月前
|
缓存 Java 数据库连接
使用MyBatis缓存的简单案例
MyBatis 是一种流行的持久层框架,支持自定义 SQL 执行、映射及复杂查询。本文介绍了如何在 Spring Boot 项目中集成 MyBatis 并实现一级和二级缓存,以提高查询性能,减少数据库访问。通过具体的电商系统案例,详细讲解了项目搭建、缓存配置、实体类创建、Mapper 编写、Service 层实现及缓存测试等步骤。
|
2月前
|
缓存 Cloud Native 安全
探索阿里巴巴新型ORM框架:超越MybatisPlus?
【10月更文挑战第9天】在Java开发领域,Mybatis及其增强工具MybatisPlus长期占据着ORM(对象关系映射)技术的主导地位。然而,随着技术的发展,阿里巴巴集团推出了一种新型ORM框架,旨在提供更高效、更简洁的开发体验。本文将对这一新型ORM框架进行探索,分析其特性,并与MybatisPlus进行比较。
45 0
|
4月前
|
Java 数据库连接 Spring
后端框架入门超详细 三部曲 Spring 、SpringMVC、Mybatis、SSM框架整合案例 【爆肝整理五万字】
文章是关于Spring、SpringMVC、Mybatis三个后端框架的超详细入门教程,包括基础知识讲解、代码案例及SSM框架整合的实战应用,旨在帮助读者全面理解并掌握这些框架的使用。
后端框架入门超详细 三部曲 Spring 、SpringMVC、Mybatis、SSM框架整合案例 【爆肝整理五万字】