Mybatis缓存理解

缓存

接触过hibernate的同学都知道hibernate中有个二级缓存,mybatis也存在二级缓存。mybatis中二级缓存有什么作用呢?mybatis提供查询缓存,可以提高查询效率,减轻数据库的压力。

一级缓存

一级缓存是session级别的缓存,基于PerpetualCache的HashMap本地存储,其存储作用域为Session,不能跨不同的session,当Session flush或close之后,该Session中的所有Cache就将清空。

SqlSession session = sqlSessionFactory.openSession();
FootballMatchMapper footballMatchMapper = session.getMapper(FootballMatchMapper.class);
FootballMatch footballMatch = footballMatchMapper.selectByPrimaryKey("20180115004");
System.out.println(footballMatch.getHomeTeamName());

//重复查询同一条记录时,一级缓存开始使用
footballMatch = footballMatchMapper.selectByPrimaryKey("20180115004");
System.out.println(footballMatch.getHomeTeamName());

//session关闭,session中的cache将被清除
session.close();
session = sqlSessionFactory.openSession();
footballMatchMapper = session.getMapper(FootballMatchMapper.class);
footballMatch = footballMatchMapper.selectByPrimaryKey("20180115004");
System.out.println(footballMatch.getHomeTeamName());

footballMatch = footballMatchMapper.selectByPrimaryKey("20180115004");
System.out.println(footballMatch.getHomeTeamName());

//sqlSession执行insert、update、delete等操作commit提交后会清空缓存区域
footballMatch.setUpdateTime(new Date());
footballMatchMapper.updateByPrimaryKeySelective(footballMatch);
footballMatch = footballMatchMapper.selectByPrimaryKey("20180115004");
System.out.println(footballMatch.getHomeTeamName());

通过日志分析一下: 
同一个session作用域内,相同查询结果会从缓存中取,不会重复从数据库查询。当session关闭之后,该Session中的所有Cache就将清空,从而从数据库查询:

update操作之后会清空缓存区域,使缓存失效,从而执行数据库查询:

二级缓存

二级缓存是mapper级别的缓存,它的实现机制跟一级缓存差不多,也是基于PerpetualCache的HashMap本地存储。作用域为mapper的namespace,可以自定义存储,比如Ehcache。Mybatis的二级缓存是跨Session的,每个Mapper享有同一个二级缓存域.

Mybatis内部存储缓存使用一个HashMap,key为hashCode+sqlId+Sql语句。value为从查询出来映射生成的Java对象。

SqlSession session1 = sqlSessionFactory.openSession();
FootballMatchMapper footballMatchMapper1 = session1.getMapper(FootballMatchMapper.class);
FootballMatch footballMatch = footballMatchMapper1.selectByPrimaryKey("20180115004");
System.out.println(footballMatch.getHomeTeamName());
session1.close();

SqlSession session2 = sqlSessionFactory.openSession();
FootballMatchMapper footballMatchMapper2 = session2.getMapper(FootballMatchMapper.class);
FootballMatch footballMatch2 = footballMatchMapper2.selectByPrimaryKey("20180115004");
System.out.println(footballMatch2.getHomeTeamName());
session2.close();

如果两个mapper文件的命名空间一样,他们就可以共享一个mapper缓存。每次查询会先从缓存区域找,如果找不到从数据库查询,查询到数据将数据写入缓存。

开启缓存

二级缓存是需要配置来开启的:

<setting name="cacheEnabled" value="true"/>

非常简单,在Mybatis的核心配置文件中加上上面这行代码,然后在Mapper映射文件中添加一行:

<cache eviction="FIFO" flushInterval="60000" size="512" readOnly="true"/>

表示开启二级缓存,flushInterval(刷新间隔)可以被设置为任意的正整数,而且它们代表一个合理的毫秒形式的时间段。默认情况是不设置,也就是没有刷新间隔,缓存仅仅调用语句时刷新。 size(引用数目)可以被设置为任意正整数,要记住你缓存的对象数目和你运行环境的可用内存资源数目。默认值是1024。 readOnly(只读)属性可以被设置为true或false。

只读的缓存会给所有调用者返回缓存对象的相同实例。因此这些对象不能被修改。这提供了很重要的性能优势。可读写的缓存会返回缓存对象的拷贝(通过序列化)。这会慢一些,但是安全,因此默认是false。

如果已经开启二级缓存的mapper里面某个查询不需要缓存,可以使用useCache=”false”禁用二级缓存:

<select id="selectByPrimaryKey" resultMap="BaseResultMap" parameterType="java.lang.String" useCache="false">

刷新缓存

在mapper的同一个namespace中,如果有其他insert、update、delete操作后都需要执行刷新缓存操作,来避免脏读。

这时我们只需要设置statement配置中的flushCache=“true“属性,就会默认刷新缓存,相反如果是false就不会了。

当然,不管开不开缓存刷新功能,你要是手动更改数据库表,那都肯定不能避免脏读的发生,那就属于手贱了。

无论是一级缓存还是二级缓存,C/U/D增删改操作commit提交后会清空缓存区域,使缓存失效。合理利用二级缓存可以提高系统性能,减少数据库压力。但是,如果使用不当可那个会出现缓存一致性问题,对查询结果实时性要求不高,此时可采用mybatis二级缓存技术降低数据库访问量,提高访问速度。

本文转自:https://blog.csdn.net/chengbinbbs/article/details/79215783

原文地址:https://www.cnblogs.com/nizuimeiabc1/p/9412864.html

时间: 2024-08-30 09:02:37

Mybatis缓存理解的相关文章

《深入理解mybatis原理4》 MyBatis缓存机制的设计与实现

<深入理解mybatis原理> MyBatis缓存机制的设计与实现 本文主要讲解MyBatis非常棒的缓存机制的设计原理,给读者们介绍一下MyBatis的缓存机制的轮廓,然后会分别针对缓存机制中的方方面面展开讨论. MyBatis将数据缓存设计成两级结构,分为一级缓存.二级缓存:   一级缓存是Session会话级别的缓存,位于表示一次数据库会话的SqlSession对象之中,又被称之为本地缓存.一级缓存是MyBatis内部实现的一个特性,用户不能配置,默认情况下自动支持的缓存,用户没有定制它

mybatis的sql 缓存,去除mybatis缓存

第二次用到mybaits,还是被同一个问题坑了几个小时,所以一定要把这个问题分享给大家.网友很多都是说了一大堆的配置,都是在mybatis.xml中配置.但是,我是没有用mybatis.xml的,我的配置都是在每个**mapper.xml中配置,无非就是一个关联实体类的路径,和一个别名. 在mybatis框架中,在SqlSession未关闭之前,在一个session里面,如果执行相同的select语句,mybatis不会重新查询数据库,而是直接返回缓存在内存中的查询结果,这个是与MyBatis的

MyBatis缓存设计

和大多数ORM框架一样,为了尽可能减少数据库的访问,MyBatis设计支持缓存功能.设计上通过Cache接口提供SPI(服务提供接口),可以让第三方缓存提供具体的缓存实现,比如使用ehcache.Redis等,通过这些常用环境提供的实现类,即可将其配置成MyBatis中使用的缓存. 除了给第三方缓存提供SPI接口外,MyBatis自身也通过HashMap做了简单的缓存实现.总体上MyBatis缓存设计分为一级缓存和二级缓存. MyBatis一级缓存是SqlSession级别的本地缓存,默认开启不

MyBatis缓存通俗易懂

1.1     mybatis缓存介绍 如下图,是mybatis一级缓存和二级缓存的区别图解: Mybatis一级缓存的作用域是同一个SqlSession,在同一个sqlSession中两次执行相同的sql语句,第一次执行完毕会将数据库中查询的数据写到缓存(内存),第二次会从缓存中获取数据将不再从数据库查询,从而提高查询效率.当一个sqlSession结束后该sqlSession中的一级缓存也就不存在了.当遇到增删改时会清空缓存.Mybatis默认开启一级缓存. Mybatis二级缓存是多个Sq

【转】MaBatis学习---源码分析MyBatis缓存原理

[原文]https://www.toutiao.com/i6594029178964673027/ 源码分析MyBatis缓存原理 1.简介 在 Web 应用中,缓存是必不可少的组件.通常我们都会用 Redis 或 memcached 等缓存中间件,拦截大量奔向数据库的请求,减轻数据库压力.作为一个重要的组件,MyBatis 自然也在内部提供了相应的支持.通过在框架层面增加缓存功能,可减轻数据库的压力,同时又可以提升查询速度,可谓一举两得.MyBatis 缓存结构由一级缓存和二级缓存构成,这两级

mybatis缓存的设计

继续用提问的方式来看Mybatis的缓存设计. 1.Mybatis如何开启缓存 Mybatis对查询结果进行缓存,所以缓存的对象为具体的Statement 通过在Statement上是否使用缓存来启用. <select id="selectByPrimaryKey" parameterType="java.lang.Long" resultMap="BaseResultMap" fetchSize="5" stateme

关于 Mybatis 缓存机制,面试官都未必知道的这么详细

欢迎关注个人微信公众号: 小哈学Java, 文末分享阿里 P8 高级架构师吐血总结的 <Java 核心知识整理&面试.pdf>资源链接!! 个人网站: https://www.exception.site 一.前言 MyBatis是常见的Java数据库访问层框架.在日常工作中,开发人员多数情况下是使用MyBatis的默认缓存配置,但是MyBatis缓存机制有一些不足之处,在使用中容易引起脏数据,形成一些潜在的隐患.个人在业务开发中也处理过一些由于MyBatis缓存引发的开发问题,带着个

聊聊MyBatis缓存机制

前言 MyBatis是常见的Java数据库访问层框架.在日常工作中,开发人员多数情况下是使用MyBatis的默认缓存配置,但是MyBatis缓存机制有一些不足之处,在使用中容易引起脏数据,形成一些潜在的隐患.个人在业务开发中也处理过一些由于MyBatis缓存引发的开发问题,带着个人的兴趣,希望从应用及源码的角度为读者梳理MyBatis缓存机制. 本次分析中涉及到的代码和数据库表均放在GitHub上,地址: mybatis-cache-demo .欢迎大家关注我的主页,给大家分享我的经验以及架构资

Mybatis(四) Mybatis缓存

4.1 Mybatis缓存概念 ??缓存就是内存中的数据,常常来自对数据库查询结果的保存,使用缓存,我们可以避免频繁的与数据进行交互,进而提高响应速度.Mybatis 也提供了对缓存的支持,分为一级缓存和二级缓存,通过下图来理解: 一级缓存是SqlSession级别的缓存.在操作数据库时需要构造sqlSession对象,在对象中有一个数据结构(HashMap)用于存储缓存数据.不同的sqlSession2直接的缓存区域(HashMap)是互相不影响. 二级缓存是Mapper级别的缓存,多个Sql