MyBatis中一级缓存和二级缓存的区别

 更新时间:2024年07月05日 09:36:18   作者:辞暮尔尔-烟火年年  
MyBatis提供了两级缓存机制,一级缓存和二级缓存,本文主要介绍了MyBatis中一级缓存和二级缓存的区别,具有一定的参考价值,感兴趣的可以了解一下

MyBatis 提供了两级缓存机制:一级缓存(Session级别)和二级缓存(全局级别),以提高应用的性能通过减少数据库的查询次数。

一级缓存(Session级别)

一级缓存是基于 SQL 会话(SqlSession)的,它是默认开启的。一级缓存的生命周期与 SQL 会话一致,当会话结束时,缓存也随之消失。这意味着,在同一个 SQL 会话中,对于相同的查询请求,第一次会从数据库中获取数据,并缓存结果;之后相同的查询请求将直接从缓存中获取数据,不会再去访问数据库。

try (SqlSession session = sqlSessionFactory.openSession()) {
    BlogMapper mapper = session.getMapper(BlogMapper.class);
    
    // 第一次查询,数据来自数据库
    Blog blog1 = mapper.selectBlogById(1);
    // 第二次查询,相同的会话中,数据来自一级缓存
    Blog blog2 = mapper.selectBlogById(1);
}

一级缓存主要是利用了一个 Map 来存储缓存数据,关键代码在 DefaultSqlSession 类中的 selectList 方法里,它使用了 Executor 的 query 方法来实现。

public <E> List<E> selectList(String statement, Object parameter, RowBounds rowBounds) {
    try {
        MappedStatement ms = configuration.getMappedStatement(statement);
        return executor.query(ms, wrapCollection(parameter), rowBounds, Executor.NO_RESULT_HANDLER);
    } catch (Exception e) {
        throw ExceptionFactory.wrapException("Error querying database.  Cause: " + e, e);
    } finally {
        ErrorContext.instance().reset();
    }
}

在 CachingExecutor 类的 query 方法中实现了一级缓存的逻辑:

public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException {
    BoundSql boundSql = ms.getBoundSql(parameter);
    CacheKey key = createCacheKey(ms, parameter, rowBounds, boundSql);
    return query(ms, parameter, rowBounds, resultHandler, key, boundSql);
}

二级缓存(全局级别)

二级缓存是跨 SQL 会话的,它基于 namespace 级别。开启二级缓存后,数据会存储在全局作用域内,这意味着,即使 SQL 会话关闭,缓存数据仍然可用,可以被其他 SQL 会话复用。

要启用二级缓存,需要在 MyBatis 配置文件中添加相应的配置,并在映射文件中明确指定哪些映射器使用二级缓存。

<!-- 在 mybatis-config.xml 中启用全局二级缓存 -->
<configuration>
    <settings>
        <setting name="cacheEnabled" value="true"/>
    </settings>
</configuration>

在映射文件中使用二级缓存:

<!-- 在 Mapper.xml 中开启二级缓存 -->
<cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/>

二级缓存的关键实现在 CachingExecutor 类:

public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler,
                         CacheKey key, BoundSql boundSql) throws SQLException {
    Cache cache = ms.getCache();
    if (cache != null) {
        flushCacheIfRequired(ms);
        if (ms.isUseCache() && resultHandler == null) {
            ensureNoOutParams(ms, boundSql);
            @SuppressWarnings("unchecked")
            List<E> list = (List<E>) tcm.getObject(cache, key);
            if (list == null) {
                list = delegate.query(ms, parameter, rowBounds, resultHandler, key, boundSql);
                tcm.putObject(cache, key, list); // issue #578 and #116
            }
            return list;
        }
    }
    return delegate.query(ms, parameter, rowBounds, resultHandler, key, boundSql);
}

区别总结

  • 作用范围:一级缓存是基于 SqlSession 的,其生命周期也随之绑定;而二级缓存是基于 Mapper 的 namespace,对整个应用有效。
  • 生命周期:一级缓存随着 SQL 会话的结束而失效,二级缓存则可以跨会话使用。
  • 可自定义性:二级缓存提供了更多的自定义设置,例如失效策略、大小限制等,通过在 Mapper.xml 中配置 <cache> 标签来实现。
  • 数据安全:二级缓存由于是跨会话的,使用时需要更加注意数据的一致性问题。在使用二级缓

到此这篇关于MyBatis中一级缓存和二级缓存的区别的文章就介绍到这了,更多相关MyBatis 一级缓存和二级缓存内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家! 

相关文章

  • Java中的自旋锁与阻塞锁详解

    Java中的自旋锁与阻塞锁详解

    这篇文章主要介绍了Java中的自旋锁与阻塞锁详解,阻塞锁是指当线程尝试获取锁失败时,线程进入阻塞状态,直到接收信号后被唤醒,阻塞或者唤醒一个Java线程需要操作系统切换CPU 状态来完成,这种状态转换 需要耗费处理器时间,需要的朋友可以参考下
    2023-10-10
  • Java多态的使用注意事项

    Java多态的使用注意事项

    本文讲解了什么是JAVA多态和Java多态是如何实现的,在使用Java多态时需要注意什么,具体大家看下面的内容
    2013-11-11
  • 一篇文章弄懂Spring MVC的参数绑定

    一篇文章弄懂Spring MVC的参数绑定

    这篇文章主要给大家介绍了关于如何通过一篇文章弄懂Spring MVC的参数绑定,文中通过示例代码以及图文介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-02-02
  • 浅谈ThreadLocal为什么会内存泄漏

    浅谈ThreadLocal为什么会内存泄漏

    这篇文章主要介绍了浅谈ThreadLocal为什么会内存泄漏,每个Thread内部维护着一个ThreadLocalMap,它是一个Map,这个映射表的Key是一个弱引用,其实就是ThreadLocal本身,Value是真正存的线程变量Object,需要的朋友可以参考下
    2023-12-12
  • SSM框架通过mybatis-generator自动生成代码(推荐)

    SSM框架通过mybatis-generator自动生成代码(推荐)

    这篇文章主要介绍了SSM框架通过mybatis-generator自动生成代码,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2017-11-11
  • Springboot导入本地jar后 打包依赖无法加入的解决方案

    Springboot导入本地jar后 打包依赖无法加入的解决方案

    这篇文章主要介绍了Springboot导入本地jar后 打包依赖无法加入的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-11-11
  • 图解Java排序算法之堆排序

    图解Java排序算法之堆排序

    这篇文章主要为大家详细介绍了Java经典排序算法之堆排序,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-11-11
  • JavaWeb JDBC + MySql 通讯录实现简单的增删改查功能案例详解

    JavaWeb JDBC + MySql 通讯录实现简单的增删改查功能案例详解

    这篇文章主要介绍了JavaWeb JDBC + MySql 通讯录实现简单的增删改查功能,结合具体案例形式详细分析了JavaWeb JDBC + MySql数据库连接、增删改查等相关操作技巧与注意事项,需要的朋友可以参考下
    2019-08-08
  • 八种Java中的基本数据类型详解

    八种Java中的基本数据类型详解

    在Java编程中,基本数据类型是必不可少的一部分,对于初学者而言,理解这些基本数据类型是非常重要的,下面我们就来学习一下Java中的八种基本数据类型,以及它们的使用方法吧
    2023-08-08
  • spring-boot react如何一步一步实现增删改查

    spring-boot react如何一步一步实现增删改查

    这篇文章主要介绍了spring-boot react如何一步一步实现增删改查,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-11-11

最新评论