Mybatis 缓存

本文详细介绍了Mybatis的一级缓存和二级缓存机制,包括缓存的作用、工作原理、配置及自定义缓存。一级缓存是SqlSession级别的,而二级缓存是基于namespace,可配置缓存策略,如LRU、FIFO等。通过缓存,可以减少数据库交互,提升高并发场景下的应用性能。文章还展示了如何在实践中使用和配置Mybatis缓存,以及需要注意的缓存失效情况。
摘要由CSDN通过智能技术生成

Mybatis 缓存

我们知道当我们需要去查数据库的时候,都需要与数据库产生连接,然后关闭连接,但是这种动作是十分浪费资源的,这个时候就出现了缓存机制。

缓存机制就是用来解决高并发问题的。

早期的时候网络用户不多的时候,只需要客户端访问服务器,服务器去数据库查询就可以查出来了

请添加图片描述

但是当后来用户越来越多的时候,访问速度开始越来越慢,然后就出现了Memcached缓存服务器,将一些查询出来的东西放在缓存里面,这样在下次查询相同数据的时候,就不用走数据库了,直接走Memcached缓存服务器了

请添加图片描述

使用缓存能够让我们减少与数据库的交互,来解决高并发问题。

注意点:

  • 建议在经常查询且不经常改变的数据上使用缓存。

  • 映射语句文件中的所有 select 语句的结果将会被缓存。

  • 映射语句文件中的所有 insert、update 和 delete 语句会刷新缓存。

  • 缓存会使用最近最少使用算法(LRU, Least Recently Used)算法来清除不需要的缓存。

    • LRU – 最近最少使用:移除最长时间不被使用的对象。
    • FIFO – 先进先出:按对象进入缓存的顺序来移除它们。
    • SOFT – 软引用:基于垃圾回收器状态和软引用规则移除对象。
    • WEAK – 弱引用:更积极地基于垃圾收集器状态和弱引用规则移除对象。
  • 缓存不会定时进行刷新(也就是说,没有刷新间隔)。

缓存失效的情况:

  • 查询不同的结果
  • 使用增删改语句进行操作时会刷新数据库
  • 查询不同的Mapper.xml
  • 手动清除缓存

一级缓存

一级缓存是默认开启的,它是SqlSession级别的,一次会话内有效,也就是在SqlSession的开启和关闭期间才有效。

@Test
    public void test(){
        SqlSession sqlSession = MybatisUtils.getSqlSession();
        BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
        HashMap map = new HashMap();

        ArrayList<Integer> ids = new ArrayList<>();
        ids.add(1);

        map.put("ids",ids);
        mapper.getBlogForeach(map);

        mapper.getBlogForeach(map);

        sqlSession.close();
    }

我们查询相同的一个人,这个人的id都为1

请添加图片描述

可以看到我们只执行了一次SQL,这就是缓存

但是当我们查询不同的用户的时候是需要执行两次SQL的

@Test
    public void test(){
        SqlSession sqlSession = MybatisUtils.getSqlSession();
        BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
        HashMap map = new HashMap();

        ArrayList<Integer> ids = new ArrayList<>();
        ids.add(1);

        map.put("ids",ids);
        mapper.getBlogForeach(map);

        System.out.println("------------------------------------------");

        ids.remove(0);
        ids.add(2);
        mapper.getBlogForeach(map);

        sqlSession.close();
    }

可以看到结果执行了两条SQL

请添加图片描述

二级缓存

默认情况下,Mybatis只启动了会话等级的缓存也就是一级缓存,而二级缓存是需要手动开启和配置的。

二级缓存是基于namespace级别的,一个命名空间对应一个二级缓存。

工作机制

  • 一个会话查询的数据会被保存在一级缓存中
  • 但是当这个会话关闭的时候,一级缓存中的数据也就不存在了,但是我们想要的是,会话关闭之后,一级缓存的数据会被保存到二级缓存
  • 这样新的会话查询信息的时候就可以从二级缓存中获取数据了
  • 不同的Mapper查出的数据会存在自己对应的二级缓存中

使用:

<!--    开启二级缓存    -->
<cache/>

你也可以手动的去定制二级缓存

<cache
    <!--   清楚策略     -->
            eviction="FIFO"
    <!--    自动刷新缓存的时间间隔,以毫秒为单位,下面是60秒    -->
            flushInterval="60000"
    <!--    最多存储512个引用    -->
            size="512"
    <!--    缓存中的数据只读    -->
            readOnly="true"/>

下面写了一个测试类,查询指定id的用户,可以看到我们先关闭了sqlSession,然后使用sqlSession2去查询

@Test
    public void test(){
        SqlSession sqlSession = MybatisUtils.getSqlSession();
        SqlSession sqlSession2 = MybatisUtils.getSqlSession();
        BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
        BlogMapper mapper2 = sqlSession2.getMapper(BlogMapper.class);
        HashMap map = new HashMap();

        ArrayList<Integer> ids = new ArrayList<>();
        ids.add(1);

        map.put("ids",ids);
        mapper.getBlogForeach(map);
        sqlSession.close();



        HashMap map2 = new HashMap();

        ArrayList<Integer> ids2 = new ArrayList<>();
        ids2.add(1);

        map2.put("ids",ids2);
        mapper2.getBlogForeach(map2);

        sqlSession2.close();
    }

保存到缓存中的对象实体类必须实现序列化接口Serializable,否则就会报错

请添加图片描述

最终结果

请添加图片描述

小结

  • 每一个Mapper查询的数据都只会存在自己对应的二级缓存中
  • 所有的缓存都会先放在一级缓存中
  • 只有在一级缓存失效的时候才会转存到二级缓存

Mybatis缓存原理

请添加图片描述

自定义缓存

我们可以自己定制一些缓存,使用方法如下

1.导包

<!-- https://mvnrepository.com/artifact/org.mybatis.caches/mybatis-ehcache -->
<dependency>
    <groupId>org.mybatis.caches</groupId>
    <artifactId>mybatis-ehcache</artifactId>
    <version>1.2.2</version>
</dependency>

2.开启缓存

<cache type="org.mybatis.caches.ehcache.EhcacheCache"/>

3.自定义缓存配置文件

<?xml version="1.0" encoding="UTF-8"?>
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd"
         updateCheck="false">
    
    <diskStore path="./tmpdir/Tmp_EhCache"/>

    <defaultCache
            eternal="false"
            maxElementsInMemory="10000"
            overflowToDisk="false"
            diskPersistent="false"
            timeToIdleSeconds="1800"
            timeToLiveSeconds="259200"
            memoryStoreEvictionPolicy="LRU"/>

    <cache
            name="cloud_user"
            eternal="false"
            maxElementsInMemory="5000"
            overflowToDisk="false"
            diskPersistent="false"
            timeToIdleSeconds="1800"
            timeToLiveSeconds="1800"
            memoryStoreEvictionPolicy="LRU"/>
</ehcache>
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值