MyBatis缓存机制 ▎特殊符号处理

前言:

MyBatis 的缓存机制通过一级缓存和二级缓存显著提升系统性能。一级缓存在 SQL 会话中减少重复查询,二级缓存跨会话共享查询结果,但引入了数据一致性和内存管理的挑战。开发者需平衡性能与数据准确性。此外,MyBatis 还提供了多种方法来安全处理 SQL 中的特殊符号,防止 SQL 注入,并灵活应用动态 SQL。这些功能对于优化性能和确保数据安全至关重要

特殊符号处理

MyBatis中的sql是在xml文件中编写的,所以我们就不可以做直接编写 >  < 等特殊符号

解决方法一:  <   对应: &lt    > 对应: &gt   '' 对应: &quot   , 对应: &apos  & 对应: &amp    

解决方法二:将特殊符号书写<![CDATA[特殊符号]]>   将特殊符号填写在第二个中括号中

MyBatis缓存机制

什么是缓存?

缓存 (Cache):是一种用于临时存储数据的技术,将数据临时存储在内存中

缓存作用

缓存(cache)的作用是为了减去数据库的压力,提高查询性能。缓存实现的 原理是从数据库中查询出来的对象在使用完后不要销毁,而是存储在内存(缓存) 中,当再次需要获取该对象时,直接从内存(缓存)中直接获取,不再向数据库 执行 select 语句,从而减少了对数据库的查询次数,因此提高了数据库的性能

缓存机制

如果有缓存查询流程,先从缓存中查数据,如果缓存中没有,去数据库查询,查询到后把数据放到缓存中,下次直接从缓存中获取

MyBatis一级缓存

   Mybatis以及缓存,默认是sqlSession级别的,再同一个SqlSession中查询到数据先缓存SqlSession对象中,第二次查询数据时,先从SqlSession中查询,如果有直接返回,没有,再去数据库查

一级缓存生命周期:

   开始于:SqlSession创建

   结束于:

  1. SqlSession关闭 ----->  sqlSession.close()
  2. 如果期间执行了新增,修改,删除操作,也会清空SqlSession对象中的缓存
  3. 调用SqlSession.cleanCache()强制清空一级缓存

代码演示解释缓存:

    public void find() {
        SqlSession sqlSession = MyBatisUtil.getSqlSession();
        TeacherDao teacherDao = sqlSession.getMapper(TeacherDao.class);
        Teacher teacher1 = teacherDao.findTeacherById(4);
        Teacher teacher2 = teacherDao.findTeacherById(4);
        Teacher teacher3 = teacherDao.findTeacherById(4);
        Teacher teacher4 = teacherDao.findTeacherById(4);
        System.out.println(teacher1);
        System.out.println(teacher2);
        System.out.println(teacher3);
        System.out.println(teacher4);
        sqlSession.close();
    }

我们这里多次查询teacher,然后放在不同的对象中

结果:

画图解释:

MyBatis二级缓存

  二级缓存是SqlSessionFactory级别,可以让多个SqlSession共享数据

  mybatis默认没有开启二级缓存,使用时需要配置开启

  如果开启了二级缓存,当SqlSession关闭时,会将一级缓存的数据缓存到二级缓存中,其他的SqlSession就可以从二级缓存中查询到之前的SqlSession缓存的数据

如何开启二级缓存

第一步:在MyBatis配置文件中配置

    <settings>
        <setting name="logImpl" value="STDOUT_LOGGING"/>
        <setting name="mapUnderscoreToCamelCase" value="true"/>
<!--        <setting name="autoMappingBehavior" value="FULL"/> &lt;!&ndash;设置多张表自动映射-->
        <setting name="cacheEnabled" value="true"/> <!--开启二级缓存-->
    </settings>

第二步:配置映射文件

在 Mapper 映射文件中添加<cache />,表示此 mapper 开启二级缓存. 当 SqlSeesion 关闭时,会将数据存入到二级缓存

第三步:对象序列化

被缓存的 POJO(实体类)必须实现序列化接口(implements Serializable)

代码演示:

    public void find() {
        SqlSession sqlSession = MyBatisUtil.getSqlSession();
        TeacherDao teacherDao = sqlSession.getMapper(TeacherDao.class);
        Teacher teacher = new Teacher();
        teacher =  teacherDao.findTeacherById(4);
        System.out.println(teacher);
        sqlSession.close();

        SqlSession sqlSession1 = MyBatisUtil.getSqlSession();
        TeacherDao teacherDao1 = sqlSession1.getMapper(TeacherDao.class);
        Teacher teacher1 = new Teacher();
        teacher1 =  teacherDao1.findTeacherById(4);
        System.out.println(teacher1);
        sqlSession.close();
    }

二级缓存优缺点 

优点:

  1. 提高性能

    • 二级缓存可以显著减少数据库访问的次数,尤其是在相同的数据被多次查询的情况下,从而提高应用的整体性能。
  2. 减少数据库压力

    • 通过缓存经常使用的数据,可以减少对数据库的压力,减少数据库的负载和响应时间。
  3. 灵活配置

    • MyBatis 的二级缓存支持自定义实现,你可以根据具体需求调整缓存的存储方式、过期时间等参数。
  4. 缓存隔离

    • 每个 MyBatis 的映射器(Mapper)都拥有独立的二级缓存,这样可以防止不同模块之间的缓存数据相互干扰,提高了缓存的可靠性。

缺点:

  1. 数据一致性问题

    • 缓存数据可能变得陈旧(stale),尤其是在涉及多方更新的情况下,可能导致缓存中的数据和数据库中的实际数据不同步,出现数据不一致的问题。
  2. 额外的内存开销

    • 二级缓存会占用一定的内存资源,缓存的数据越多,占用的内存也越多。如果不加以控制,可能导致内存压力增大。
  3. 配置复杂度

    • 对于初学者或项目复杂性较高的情况,正确配置二级缓存可能需要更多的经验和时间。
  4. 无法缓存多表联合查询结果

    • 二级缓存通常只能缓存单表查询的结果,复杂的多表联合查询结果缓存效果不佳。
  5. 需要注意缓存失效机制

    • 如果缓存未正确失效,可能会导致数据不一致,因此需要精心设计缓存的失效策略。

<cache>标签

1. type

  • 解释:指定缓存的实现类,通常用于自定义缓存。
  • 默认值org.apache.ibatis.cache.impl.PerpetualCache
  • 示例<cache type="com.example.MyCustomCache"/>

2. eviction

  • 解释:指定缓存的回收策略,也就是缓存满了之后,如何清除旧的数据。常用策略有:
    • LRU (Least Recently Used):最近最少使用的策略,移除最久未被使用的对象。
    • FIFO (First In, First Out):先进先出策略,按照进入缓存的顺序来移除对象。
    • SOFT:软引用,基于垃圾回收器的回收情况来移除对象。
    • WEAK:弱引用,当垃圾回收器运行时,总是会移除。
  • 默认值LRU
  • 示例<cache eviction="FIFO"/>

3. flushInterval

  • 解释:指定缓存的刷新间隔时间(单位是毫秒),也就是多长时间后缓存将自动刷新。如果不设置,缓存不会自动刷新,直到某个条件触发时才会失效。
  • 默认值:没有默认值(即缓存不会自动刷新)。
  • 示例<cache flushInterval="60000"/>(缓存每分钟刷新一次)

4. size

  • 解释:指定缓存的大小,也就是缓存中可以存放多少条查询结果。在超出这个大小时,将按照 eviction 策略进行缓存回收。
  • 默认值:没有默认值(缓存大小无限制,但取决于内存容量)。
  • 示例<cache size="512"/>(最多缓存 512 条查询结果)

5. readOnly

  • 解释:指定缓存的数据是否只读。只读缓存可以提高性能,因为 MyBatis 可以在多个线程之间安全地共享这些缓存对象,但它们不能被修改。如果设置为 false,则缓存中的对象是可变的,且不会被多个线程共享。
  • 默认值false
  • 示例<cache readOnly="true"/>

6. blocking

  • 解释:指定是否启用阻塞缓存。在阻塞缓存模式下,当某个缓存键值的数据正在加载时,其他请求将被阻塞,直到加载完成。
  • 默认值false
  • 示例<cache blocking="true"/>

 



 感谢大家的观看,本次分享就到这里。希望我的内容能够对您有所帮助。创作不易,欢迎大家多多支持,您的每一个点赞都是我持续更新的最大动力!如有不同意见,欢迎在评论区积极讨论,让我们一起学习、共同进步!如果有相关问题,也可以私信我,我会认真查看每一条留言。期待下次再见!

                                       希望路飞的笑容可以治愈努力路途中的你我!

博主vx:Dreamkid05 --->欢迎大家和博主讨论问题 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值