课时10:MyBatis一级缓存、二级缓存

.1)查询缓存

  1.一级缓存

    1.1 同一个SqlSession对象,mybatis默认就开启了一级缓存,下方为示意图

    1.2 如果用同样的SqlSession对象查询相同的数据,则只会在第一次 查询时 向数据库发送SQL语句,并将查询的结果 放入SqlSession中(作为缓存存在);后续再次查询同样的对象时,则直接从缓存中查询该对象即可(则省略了数据库的访问)

    1.3 一旦commit就会清空SqlSession中的缓存

    1.4测试代码如下

复制代码
 public static void selectOneStudent() throws IOException {
        Reader reader = Resources.getResourceAsReader("config.xml");
        SqlSessionFactory sessionFactory=new SqlSessionFactoryBuilder().build(reader);
        SqlSession session=sessionFactory.openSession();
        IStudentDao iStudentDao=session.getMapper(IStudentDao.class);
        Student student = iStudentDao.selectStudentById(1);
        session.commit();
        Student student2 = iStudentDao.selectStudentById(1);
        System.out.println(student);
        System.out.println(student2);
        session.close();
        reader.close();
    }
复制代码

  2.二级缓存

    2.1 mybatis默认关闭二级缓存 需要手动的打开

      2.1.1 在主配置文件中配置

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

      2.1.2 在具体的mapper.xml声明开启

<!--    声明开启-->
<cache/>

      2.1.3 异常提示 :java.io.NotSerializableExceptio可知,mybatis的二级缓存是将对象 放入硬盘

        序列化:内存---》硬盘

        反序列化:硬盘---》内存

        如何解决?

         在实体类中实现Serializable接口 (序列化student类,以及Student的级联属性,和Student父类)

    2.2  二级缓存的范围在同一个namespace中产生的接口生成的代理对象都可以共享缓存

 

       2.3 如果是同一个SqlSession对象进行多次查询,则直接进入一级缓存查询; 如果不是同一个SqlSession进行多次查询(但是都是一个namespace),则进入二级缓存

    2.4 测试如下:

复制代码
 public static void selectOneStudent() throws IOException {
        Reader reader = Resources.getResourceAsReader("config.xml");
        SqlSessionFactory sessionFactory=new SqlSessionFactoryBuilder().build(reader);
        SqlSession session=sessionFactory.openSession();
        IStudentDao iStudentDao=session.getMapper(IStudentDao.class);
        Student student = iStudentDao.selectStudentById(1);
        SqlSession session2=sessionFactory.openSession();
        IStudentDao iStudentDao2=session2.getMapper(IStudentDao.class);
        Student student2 = iStudentDao2.selectStudentById(1);
        System.out.println(student);
        System.out.println(student2);
        session.close();
        reader.close();
    }
复制代码

    2.5 细节补充在查询完一次之后需要关闭SqlSession,不然该对象不会存入二级缓存

      存储时机就是当执行SqlSession.close()时 才会存入到二级缓存,

      如果多个文件的namespace的值相同,通过这些mapper.xml产生的mapper对象 仍然共享二级缓存

 

posted @   何邦柱  阅读(152)  评论(1编辑  收藏  举报
编辑推荐:
· 深入理解 Mybatis 分库分表执行原理
· 如何打造一个高并发系统?
· .NET Core GC压缩(compact_phase)底层原理浅谈
· 现代计算机视觉入门之:什么是图片特征编码
· .NET 9 new features-C#13新的锁类型和语义
阅读排行:
· Sdcb Chats 技术博客:数据库 ID 选型的曲折之路 - 从 Guid 到自增 ID,再到
· 语音处理 开源项目 EchoSharp
· 《HelloGitHub》第 106 期
· Spring AI + Ollama 实现 deepseek-r1 的API服务和调用
· 使用 Dify + LLM 构建精确任务处理应用
点击右上角即可分享
微信分享提示