Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >mybatis二级缓存的作用范围_java缓存机制

mybatis二级缓存的作用范围_java缓存机制

作者头像
全栈程序员站长
发布于 2022-11-05 09:11:29
发布于 2022-11-05 09:11:29
64800
代码可运行
举报
运行总次数:0
代码可运行

大家好,又见面了,我是你们的朋友全栈君。

应用场景: 对于访问多的查询请求且用户对查询结果实时性要求不高,此时可采用mybatis二级缓存技术降低数据库访问量,提高访问速度,业务场景比如:耗时较高的统计分析sql、电话账单查询sql等。 实现方法如下:通过设置刷新间隔时间,由mybatis每隔一段时间自动清空缓存,根据数据变化频率设置缓存刷新间隔flushInterval,比如设置为30分钟、60分钟、24小时等,根据需求而定。

局限性: mybatis二级缓存对细粒度的数据级别的缓存实现不好,比如如下需求:对商品信息进行缓存,由于商品信息查询访问量大,但是要求用户每次都能查询最新的商品信息, 此时如果使用mybatis的二级缓存就无法实现当一个商品变化时只刷新该商品的缓存信息而不刷新其它商品的信息,因为mybaits的二级缓存区域以mapper为单位划分, 当一个商品信息变化会将所有商品信息的缓存数据全部清空。解决此类问题需要在业务层根据需求对数据有针对性缓存。   缓存都是实现了Cache这个接口…..

如何开启 二级缓存,步骤如下: 1.导入ehcache相关jar包 (ehcache: 缓存插件,插件:就是对现有应用软件功能的一个扩展)

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
      ehcache-core-2.6.5.jar
        mybatis-ehcache-1.1.0.jar

      <dependency>
          <groupId>net.sf.ehcache</groupId>
          <artifactId>ehcache-core</artifactId>
          <version>2.6.6</version>
      </dependency>
      <!-- https://mvnrepository.com/artifact/org.mybatis.caches/mybatis-ehcache -->
      <dependency>
          <groupId>org.mybatis.caches</groupId>
          <artifactId>mybatis-ehcache</artifactId>
          <version>1.1.0</version>
      </dependency>

2,开启mybatis的二级缓存   

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
在mybatis核心配置文件mybatis-config.xml中加入

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

3.在classpath下加入ehcache.xml文件

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="../config/ehcache.xsd">

    <diskStore path="java.io.tmpdir"/>

	<defaultCache
            maxElementsInMemory="10000"
            eternal="false"
            timeToIdleSeconds="120"
            timeToLiveSeconds="120"
            overflowToDisk="true"
            maxElementsOnDisk="10000000"
            diskPersistent="false"
            diskExpiryThreadIntervalSeconds="120"
            memoryStoreEvictionPolicy="LRU"
            />
			
	<!--
		默认缓存配置,
		以下属性是必须的:
			name :cache的标识符,在一个CacheManager中必须唯一。
			maxElementsInMemory : 在内存中缓存的element的最大数目。
			maxElementsOnDisk : 在磁盘上缓存的element的最大数目。
			eternal : 设定缓存的elements是否有有效期。如果为true,timeouts属性被忽略。
			overflowToDisk : 设定当内存缓存溢出的时候是否将过期的element缓存到磁盘上。

		以下属性是可选的:
			 timeToIdleSeconds : 缓存element在过期前的空闲时间。
			 timeToLiveSeconds : 缓存element的有效生命期。
			 diskPersistent : 在VM重启的时候是否持久化磁盘缓存,默认是false。
			 diskExpiryThreadIntervalSeconds : 磁盘缓存的清理线程运行间隔,默认是120.
			 memoryStoreEvictionPolicy : 当内存缓存达到最大,有新的element加入的时候,
				移除缓存中element的策略。默认是LRU,可选的有LFUFIFO
	-->
</ehcache>

属性说明:  diskStore:指定数据在磁盘中的存储位置。  defaultCache:当借助CacheManager.add(“demoCache”)创建Cache时,EhCache便会采用<defalutCache/>指定的的管理策略 以下属性是必须的:  maxElementsInMemory – 在内存中缓存的element的最大数目  maxElementsOnDisk – 在磁盘上缓存的element的最大数目,若是0表示无穷大  eternal – 设定缓存的elements是否永远不过期。如果为true,则缓存的数据始终有效,如果为false那么还要根据timeToIdleSeconds,timeToLiveSeconds判断  overflowToDisk – 设定当内存缓存溢出的时候是否将过期的element缓存到磁盘上 以下属性是可选的:  timeToIdleSeconds – 当缓存在EhCache中的数据前后两次访问的时间超过timeToIdleSeconds的属性取值时,这些数据便会删除,默认值是0,也就是可闲置时间无穷大  timeToLiveSeconds – 缓存element的有效生命期,默认是0.,也就是element存活时间无穷大 diskSpoolBufferSizeMB 这个参数设置DiskStore(磁盘缓存)的缓存区大小.默认是30MB.每个Cache都应该有自己的一个缓冲区.  diskPersistent – 在VM重启的时候是否启用磁盘保存EhCache中的数据,默认是false。  diskExpiryThreadIntervalSeconds – 磁盘缓存的清理线程运行间隔,默认是120秒。每个120s,相应的线程会进行一次EhCache中数据的清理工作  memoryStoreEvictionPolicy – 当内存缓存达到最大,有新的element加入的时候, 移除缓存中element的策略。默认是LRU(最近最少使用),可选的有LFU(最不常使用)和FIFO(先进先出)

4.在UserMapper.xml中开启二缓存,UserMapper.xml下的sql执行完成会存储到它的缓存区域(HashMap)

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
    <cache type="org.mybatis.caches.ehcache.EhcacheCache">
                <!-- timeToLiveSeconds 缓存自创建日期起至失效时的间隔时间 -->
                <property name="timeToIdleSeconds" value="3600" />

                <!-- timeToIdleSeconds 缓存创建以后,最后一次访问缓存的日期至失效之时的时间间隔 -->
                <property name="timeToLiveSeconds" value="3600" />

                <!-- 同ehcache参数maxElementsInMemory -->
                <property name="maxEntriesLocalHeap" value="1000" />

                <!-- 同ehcache参数maxElementsOnDisk -->
                <property name="maxEntriesLocalDisk" value="10000000" />
                
                <!-- 当内存缓存达到最大,有新的element加入的时候, 移除缓存中element的策略。默认是LRU(最近最少使用),可选的有LFU(最不常使用)和FIFO 
                    (先进先出) -->
                <property name="memoryStoreEvictionPolicy" value="LRU" />
     </cache>


            <!--mybatis ehcache缓存配置 -->
            <!-- 以下两个<cache>标签二选一,第一个可以输出日志,第二个不输出日志 -->
             <cache type="org.mybatis.caches.ehcache.LoggingEhcache" />
           
             <cache type="org.mybatis.caches.ehcache.EhcacheCache"/> 

测试:需求获取员工表所有记录数(测试二级缓存), 执行一次查询后,关闭session;然后获得一个全新的Session,再执行查询,若此时控制台未输出sql语句,且磁盘相应缓存目录下有缓存文件产生,证明二级缓存发挥了作用。 sql映射文件中:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
<cache type="org.mybatis.caches.ehcache.EhcacheCache">
        <!-- timeToLiveSeconds 缓存自创建日期起至失效时的间隔时间 -->
        <property name="timeToIdleSeconds" value="3600" />

        <!-- timeToIdleSeconds 缓存创建以后,最后一次访问缓存的日期至失效之时的时间间隔 -->
        <property name="timeToLiveSeconds" value="3600" />

        <!-- 同ehcache参数maxElementsInMemory -->
        <property name="maxEntriesLocalHeap" value="1000" />

        <!-- 同ehcache参数maxElementsOnDisk -->
        <property name="maxEntriesLocalDisk" value="10000000" />

        <!-- 当内存缓存达到最大,有新的element加入的时候, 移除缓存中element的策略。默认是LRU(最近最少使用),可选的有LFU(最不常使用)和FIFO
            (先进先出) -->
        <property name="memoryStoreEvictionPolicy" value="LRU" />
    </cache>    

dao实现类中:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 @Override
    public List<Person> selectAll() {
        //步骤:
        //通过工具类获得SqlSession的实例
        SqlSession sqlSession = MyBatisUtil.getSqlSessionInstance();

        List<Person> persons = sqlSession.selectList("com.uplooking.dao.PersonDao.selectAll");
        System.out.println("第一次查询到的员工数是:" + persons);

        //关闭sqlSession(让一级缓存失效)
        MyBatisUtil.releaseResource(sqlSession);

        //执行完下面的代码行,控制台若没有sql语句输出,证明结果来自于二级缓存,而不是重新查询的数据库。
        sqlSession = MyBatisUtil.getSqlSessionInstance();
        persons = sqlSession.selectList("com.uplooking.dao.PersonDao.selectAll");
        System.out.println("第二次查询到的员工数是:" + persons);

        //释放资源
        MyBatisUtil.releaseResource(sqlSession);

        return persons;
    }

建议:放弃二级缓存,在业务层使用可控制的缓存代替更好。 <select id=”selectUserRoles” resultType=”UserRoleVO”> select * from user_role a,role b where a.roleid = b.roleid and a.userid = #{userid} </select> 像上面这个查询,你会写到那个xml中呢?? 不管是写到RoleMapper.xml还是UserRoleMapper.xml,或者是一个独立的XxxMapper.xml中。如果使用了二级缓存,都会导致上面这个查询结果可能不正确。 如果你正好修改了这个用户的角色,上面这个查询使用缓存的时候结果就是错的。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/191453.html原文链接:https://javaforall.cn

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2022年9月20日 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
mybatis开启二级缓存小记
首先在全局配置文件 mybatis-configuration.xml 文件中加入如下代码:
Dream城堡
2018/10/09
1K0
mybatis 详解(九)------ 一级缓存、二级缓存
  上一章节,我们讲解了通过mybatis的懒加载来提高查询效率,那么除了懒加载,还有什么方法能提高查询效率呢?这就是我们本章讲的缓存。   本篇源码下载链接:http://pan.baidu.com
IT可乐
2018/01/04
9620
mybatis 详解(九)------ 一级缓存、二级缓存
MyBatis:缓存
一级缓存失效情况:没有使用到当前的一级缓存,效果就是,还需要再向数据库中发起一次查询请求!
愷龍
2023/01/13
2880
MyBatis:缓存
浅谈 MyBatis 缓存
MyBatis 默认开启了一级缓存,一级缓存是在 SqlSession 层面进行缓存的。即,同一个 SqlSession ,多次调用同一个 Mapper 和同一个方法的同一个参数,只会进行一次数据库查询,然后把数据缓存到缓冲中,以后直接先从缓存中取出数据,不会直接去查数据库。但是不同的 SqlSession 对象,因为不用的 SqlSession 都是相互隔离的,所以相同的 Mapper、参数和方法,它还是会再次发送到 SQL 到数据库去执行,返回结果。
全栈程序员站长
2022/08/31
2780
MyBatis 从入门到精通:缓存
我们每次查询数据的时候,是通过数据库查询。在数据需要大量查询时,我们就需要用到缓存,下载查询相同的时候,通过内存查询,不需要经过数据库,大大的节省的资源
默 语
2024/11/20
1180
MyBatis 从入门到精通:缓存
10. Mybatis的缓存
一级缓存是 SqlSession 级别的,通过同一个 SqlSession 查询的数据会被缓存,下次查询相同的数据,就会从缓存中直接获取,不会从数据库重新访问 , 一级缓存时默认开启的
捞月亮的小北
2023/12/01
2830
MyBatis框架:第十章:mybatis缓存
mybatis缓存 说明: 缓存指的是把一些常用的数据,保存到一个可以高速读取的缓冲区中。方便程序在频繁读取的时候,可以快速的取出数据。这就叫做缓存。
Java廖志伟
2022/09/28
2760
MyBatis框架:第十章:mybatis缓存
Mybatis的缓存
1 Mybatis的缓存: 2 Mybatis的一级缓存 : 3 Mybatis的二级缓存 二级缓存的参数配置 mybatis整合ehcache ehcache.xml文件: <ehcache
用户5927264
2019/07/31
4900
Mybatis【缓存、代理、逆向工程】
Mybatis缓存 缓存的意义 将用户经常查询的数据放在缓存(内存)中,用户去查询数据就不用从磁盘上(关系型数据库数据文件)查询,从缓存中查询,从而提高查询效率,解决了高并发系统的性能问题。 这里写图
Java3y
2018/03/15
9050
Mybatis【缓存、代理、逆向工程】
mybatis缓存之整合第三方缓存工具ehcache
1、加入以下依赖包 2、配置ehcache.xml <?xml version="1.0" encoding="UTF-8"?> <ehcache xmlns:xsi="http://www.w3.o
西西嘛呦
2020/08/26
4870
mybatis缓存之整合第三方缓存工具ehcache
缓存
一级缓存失效情况:没有使用到当前的一级缓存,效果就是,还需要再向数据库中发起一次查询请求!
后端码匠
2021/08/19
3290
缓存
【MyBatis学习13】MyBatis中的二级缓存[通俗易懂]
前面介绍了,mybatis中的二级缓存是mapper级别的缓存,值得注意的是,不同的mapper都有一个二级缓存,也就是说,不同的mapper之间的二级缓存是互不影响的。为了更加清楚的描述二级缓存,先来看一个示意图:
全栈程序员站长
2022/09/29
1.3K0
【MyBatis学习13】MyBatis中的二级缓存[通俗易懂]
如何使用ehcache作为mybatis的二级缓存?
Ehcache 是现在最流行的纯Java开源缓存框架,配置简单、结构清晰、功能强大,最初知道它,是从Hibernate的缓存开始的。
java思维导图
2018/07/26
9640
如何使用ehcache作为mybatis的二级缓存?
Data Access 之 MyBatis(五)- MyBatis Cache
MyBatis包含一个非常强大的查询缓存特性,可以非常方便的配置和定义。缓存可以极大的提高查询效率
RiemannHypothesis
2022/08/19
3390
Data Access 之 MyBatis(五)-  MyBatis Cache
Mybatis笔记二
注意:因为一个订单信息只会是一个人下的订单,所以从查询订单信息出发关联查询用户信息为一对一查询。如果从用户信息出发查询用户下的订单信息则为一对多查询,因为一个用户可以下多个订单。
HUC思梦
2020/09/03
5010
Mybatis笔记二
MyBatis-24MyBatis缓存配置【集成EhCache】
Ehcache是一个用Java实现的使用简单,高速,实现线程安全的缓存管理类库,ehcache提供了用内存,磁盘文件存储,以及分布式存储方式等多种灵活的cache管理方案。
小小工匠
2021/08/17
3290
mybatis之二级缓存
现在是资源共享的时代,同样也是知识分享的时代,如果你觉得本文能学到知识,请把知识与别人分享。
互扯程序
2018/07/30
4780
mybatis之二级缓存
图解 | 聊聊 MyBatis 缓存
MappedStatement:代表要发往数据库执行的指令,可以理解为是 SQL 的抽象表示。
悟空聊架构
2022/12/12
2600
图解 | 聊聊 MyBatis 缓存
十、MyBatis的缓存
一级缓存是SqlSession级别的,通过同一个SqlSession查询的数据会被缓存,下次查询相同的数据,就会从缓存中直接获取,不会从数据库重新访问使一级缓存。
刘大猫
2025/02/25
420
MyBatis--注解、缓存配置、其他工具
前面熟悉了MyBatis的基本使用,针对一些简单的sql语句,如对一张表的增删改查,可以使用MyBatis提供的注解,省去写映射文件的步骤,复杂的多表查询,还是建议使用映射文件的方式
aruba
2022/05/18
3210
MyBatis--注解、缓存配置、其他工具
相关推荐
mybatis开启二级缓存小记
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验