MyBatis 缓存机制

发布于 2023-12-03 16:38:01 字数 11538 浏览 42 评论 0

​ 缓存是提高软硬件系统性能的一种重要手段;硬件层面,现代先进 CPU 有三级缓存,而 MyBatis 也提供了缓存机制,通过缓存机制可以大大提高我们查询性能。

一级缓存

​ Mybatis 对缓存提供支持,但是在没有配置的默认情况下,它只开启一级缓存,一级缓存只是相对于同一个 SqlSession 而言,一级缓存又叫本地缓 存。所以在参数和 SQL 完全一样的情况下,我们使用同一个 SqlSession 对象调用一个 Mapper 方法,往往只执行一次 SQL,因为使用 SelSession 第一次查询后,MyBatis 会将其放在缓存中,以后再查询的时候,如果没有声明需要刷新,并且缓存没有超时的情况 下,SqlSession 都会取出当前缓存的数据,而不会再次发送 SQL 到数据库。

为什么要使用一级缓存,不用多说也知道个大概。但是还有几个问题我们要注意一下。

一级缓存的生命周期

  1. MyBatis 在开启一个数据库会话时,会创建一个新的 SqlSession 对象,SqlSession 对象中会有一个新的 Executor 对 象。Executor 对象中持有一个新的 PerpetualCache 对象;当会话结束时,SqlSession 对象及其内部的 Executor 对象还有 PerpetualCache 对象也一并释放掉。
  2. 如果 SqlSession 调用了 close() 方法,会释放掉一级缓存 PerpetualCache 对象,一级缓存将不可用。
  3. 如果 SqlSession 调用了 clearCache() ,会清空 PerpetualCache 对象中的数据,但是该对象仍可使用。-
  4. SqlSession 中执行了任何一个 update 操作(update()、delete()、insert()) ,都会清空 PerpetualCache 对象的数据,但是该对象可以继续使用。

如何判断两次查询是完全相同的呢

mybatis 认为,对于两次查询,如果以下条件都完全一样,那么就认为它们是完全相同的两次查询。

  • 传入的 statementId
  • 查询时要求的结果集中的结果范围
  • 这次查询所产生的最终要传递给 JDBC java.sql.Preparedstatement 的 Sql 语句字符串(boundSql.getSql() )
  • 传递给 java.sql.Statement 要设置的参数值

一级缓存的测试

public class FirstCachedTest {
    @Test
    public void test() throws Exception {
        InputStream resource = Resources.getResourceAsStream("SqlMapConfig.xml");
        SqlSessionFactory sessionFactory = new SqlSessionFactoryBuilder().build(resource);
        SqlSession session = sessionFactory.openSession();
        ArticleMapper mapper = session.getMapper(ArticleMapper.class);
        Article article1 = mapper.getArticleById(1L);
        Article article2 = mapper.getArticleById(1L);
        System.out.println(article1 == article2);  //输出 true
    }
}

执行结果:

需要注意的是,这是在单独使用 MyBatis 时进行的以及缓存测试, 如果 MyBatis 与 Spring 整合,那么 MyBatis 的一级缓存可能会失效 ,详情参见 https://blog.csdn.net/ctwy291314/article/details/81938882

关闭一级缓存

在集群部署环境下一级缓存可能也会带来问题。

假设现在有一个服务集群,有两个节点。

首先,两个节点都进行了同样的查询,两个节点都有自己的一级缓存,后续同样的查询,两个节点将不再查询数据库。

如果此时节点 1 执行了 update 语句,那么节点 1 的一级缓存会被刷新,而节点 2 的一级缓存不会改变。

为了避免这个问题,可以将一级缓存的级别设为 statement 级别的,这样每次查询结束都会清掉一级缓存。MyBatis 源码(BaseExecutor::query())如下:

在 MyBatis 的核心配置文件中,添加以下配置:

<settings>
        <setting name="localCacheScope" value="STATMENT"/>
</settings>

如果不需要全局关闭以及缓存,可以在查询时指定刷新缓存,也就是在 select 标签上添加 fluhCache=true 配置:

<select resultType="Student" flushCache="true">
    select t_password password,t_name name,sex,description from student where id = #{id}
</select>

二级缓存

​ MyBatis 的二级缓存是 Application 级别的缓存,它可以提高对数据库查询的效率,以提高应用的性能。

SqlSessionFactory 层面上的二级缓存默认是不开启的,二级缓存的开启需要进行配置, 实现二级缓存的时候,MyBatis 要求返回的 POJO 必须是可序列化的

二级缓存的配置步骤

第一步:配置 SqlMapConfig.xml(可省略)

我们实质上需要在全局配置文件中开启配置文件中的所有映射器已经配置的任何缓存,也就是 cacheEnabled 属性,但是这个属性默认值为 true,所以实际上我们可以省略该步骤。

<settings>
	<setting name="cacheEnabled" value="true"/>
</settings>

第二步:配置映射文件

cache 标签配置

若要要启用全局的二级缓存,只需要在你的 SQL 映射文件中添加一行:

<cache/>

这个简单语句的效果如下:

  • 映射语句文件中的所有 select 语句的结果将会被缓存。
  • 映射语句文件中的所有 insert、update 和 delete 语句会刷新缓存。
  • 缓存会使用最近最少使用算法(LRU, Least Recently Used)算法来清除不需要的缓存。
  • 缓存不会定时进行刷新(也就是说,没有刷新间隔)。
  • 缓存会保存列表或对象(无论查询方法返回哪种)的 1024 个引用。
  • 缓存会被视为读/写缓存,这意味着获取到的对象并不是共享的,可以安全地被调用者修改,而不干扰其他调用者或线程所做的潜在修改。
cache 标签的属性
  • eviction:设置缓存的清除策略,默认值为 LRU
    • LRU – 最近最少使用:移除最长时间不被使用的对象。
    • FIFO – 先进先出:按对象进入缓存的顺序来移除它们。
    • SOFT – 软引用:基于垃圾回收器状态和软引用规则移除对象。
    • WEAK – 弱引用:更积极地基于垃圾收集器状态和弱引用规则移除对象。
  • flushInterval:(刷新间隔)属性可以被设置为任意的正整数,设置的值应该是一个以毫秒为单位的合理时间量。 默认情况是不设置,也就是没有刷新间隔,缓存仅仅会在调用语句时刷新。
  • size:(引用数目)属性可以被设置为任意正整数,要注意欲缓存对象的大小和运行环境中可用的内存资源。默认值是 1024。
  • readOnly:(只读)属性可以被设置为 true 或 false(默认值)。只读的缓存会给所有调用者返回缓存对象的相同实例。 因此这些对象不能被修改。这就提供了可观的性能提升。 而可读写的缓存会(通过反序列化)返回缓存对象的拷贝。 速度上会慢一些,但是更安全,因此默认值是 false
SQL 语句标签配置

​ 缓存的配置和缓存实例会被绑定到 SQL 映射文件的命名空间中。 因此,同一命名空间中的所有语句和缓存将通过命名空间绑定在一起。 每条语句可以自定义与缓存交互的方式,或将它们完全排除于缓存之外,这可以通过在每条语句上使用两个简单属性来达成 。 默认情况下,语句会这样来配置:

<select ... flushCache="false" useCache="true"/>
<insert ... flushCache="true"/>
<update ... flushCache="true"/>
<delete ... flushCache="true"/>

​ 鉴于这是默认行为,显然你永远不应该以这样的方式显式配置一条语句。但如果你想改变默认的行为,只需要设置 flushCache 和 useCache 属性。比如,某些情况下你可能希望特定 select 语句的结果排除于缓存之外,或希望一条 select 语句清空缓存。类似地,你可能希望某些 update 语句执行时不要刷新缓存。

<select resultMap="articleMap" parameterType="long" useCache="true" flushCache="false">
	select * from article where id = #{id}
</select>

第三步:让实体类实现 Serializable 接口

由于 <cache/> 标签 readOnly 标签默认是 false,所以 MyBatis 在读写缓存是通过序列化与反序列化完成的。

二级缓存测试

public class FirstCachedTest {
    @Test
    public void test() throws Exception {
        InputStream resource = Resources.getResourceAsStream("SqlMapConfig.xml");
        SqlSessionFactory sessionFactory = new SqlSessionFactoryBuilder().build(resource);
        SqlSession session = sessionFactory.openSession();
        ArticleMapper mapper = session.getMapper(ArticleMapper.class);
        Article article1 = mapper.getArticleById(1L);
        session.close();
        //开启一个新的 SqlSession,在不同的 SqlSession 中才会读取二级缓存(全局缓存)
        session=sessionFactory.openSession();
        ArticleMapper mapper2 = session.getMapper(ArticleMapper.class);
        Article article2 = mapper2.getArticleById(1L);
        System.out.println(article1 == article2);
    }
}

执行结果:

二级缓存需要注意的地方

​ 对于查询多 commit 少且用户对查询结果实时性要求不高,此时采用 mybatis 二级缓存技术降低数据库访问量,提高访问速度。但不能滥用二级缓存,二 级缓存也有很多弊端,从 MyBatis 默认二级缓存是关闭的就可以看出来。二级缓存是建立在同一个 namespace 下的,如果对表的操作查询可能有多个 namespace,那么得到的数据就是错误的。

​ 举例来说:文章和标签,ArticleMapper、TagMapper。在查询文章时我们需要把文章对应的标签也查询出来,那么这个标签信息被二级缓存 在 ArticleMapper 对应的 namespace 下,这个时候有人要修改 Tag 的基本信息,那就是在 TagMapper 的 namespace 下修 改,他是不会影响到 ArticleMapper 的缓存的,那么你再次查找文章数据时,拿到的是缓存的数据,这个数据其实已经是过时的。

二级缓存数据过期问题测试

Article 和 Tag 是一对多的关系,其中 Aticle 是一,Tag 是多。

ArticleMapper

<mapper namespace="com.tjd.spring_mybatis_plus.mapper.ArticleMapper">
    <cache />
    <resultMap type="Article">
        <id column="id" property="id"></id>
        <result column="title" property="title"></result>
        <result column="content" property="content"></result>
        <collection property="tags" ofType="Tag" column="id" select="com.tjd.spring_mybatis_plus.mapper.TagMapper.getTagsByArticleId"></collection>
    </resultMap>
    <select resultMap="articleMap" parameterType="long" useCache="true">
      select * from article where id = #{id}
    </select>
</mapper>

TagMapper

<mapper namespace="com.tjd.spring_mybatis_plus.mapper.TagMapper">
    <cache/>
    <resultMap type="Tag">
        <id column="id" property="id"></id>
        <result column="content" property="content"></result>
        <association  property="article" column="article_id" javaType="Article" select="com.tjd.spring_mybatis_plus.mapper.ArticleMapper.getArticleById"></association>
    </resultMap>
    <update parameterType="Tag">
        update tag  set content=#{content} where id=#{id}
    </update>
    <select parameterType="long" resultMap="tagMap">
        select * from tag where article_id=#{article_id}
    </select>
</mapper>

测试代码

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class SecondCachedErrorTest {

    @Autowired
    private ArticleMapper articleMapper;

    @Autowired
    private TagMapper tagMapper;

    @Test
    public void test() throws IOException {
        //第一次查询,查询出来的 aticle 对象被缓存在 ArticleMapper 的 namespace 下
        Article article = articleMapper.getArticleById(1L);
        Tag tag = article.getTags().get(0);
        tag.setContent("dasdas");
        //更新 Tag,那么 TagMapper 下的二级缓存被刷新(清空)
        tagMapper.updateTag(tag);
        //再次查询 Article,此时获得是缓存数据,而关联的 tag 数据已经过时
        Article article2 = articleMapper.getArticleById(1L);
        //TagMapper 对应的 namespace 下的缓存由于在更新时被刷新(清空),所以查询的结果是正确的
        List<Tag> tags = tagMapper.getTagsByArticleId(1L);
    }
}

根据以上测试,我们明白想要使用二级缓存时需要想好两个问题

  • 对该表的操作与查询都在同一个 namespace 下,其他的 namespace 如果有操作,就会发生缓存数据过期的问题。
  • 对关联表的查询,关联的所有表的操作都必须在同一个 namespace。
  • 在有多表查询的情况下建议不使用二级缓存。

两级缓存的优先级

如果两级缓存同时开启,那么二级缓存比一级缓存优先级高,也就是在执行数据库查询操作时,优先读取二级缓存中的内容


推荐文章(美团点评团队)

文章参考至:

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

甩你一脸翔

暂无简介

0 文章
0 评论
22 人气
更多

推荐作者

qq_E2Iff7

文章 0 评论 0

Archangel

文章 0 评论 0

freedog

文章 0 评论 0

Hunk

文章 0 评论 0

18819270189

文章 0 评论 0

wenkai

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文