缓存在互联网系统中是非常重要的, 其主要作用是将数据保存到内存中, 当用户查询数据 时, 优先从缓存容器中获取数据,而不是频繁地从数据库中查询数据,从而提高查询性能。目 前流行的缓存服务器有MongoDB 、Redis 、Ehcache 等,不同的缓存服务器有不同的应用场景, 不存在孰优孰劣。 ​ MyBatis 提供一级缓存和二级缓存的机制。一级缓存是SqlSession 级别的缓存, 在操作数据 库时, 每个SqlSession 类的实体对象中都有一个HashMap 数据结构可以用来缓存数据,不同的 SqlSession 类的实例对象缓存的HashMap 数据结构互不影响。二级缓存是Mapper 级别的缓存, 多个SqlSession 类的实例对象操作同一个Mapper 配置文件中的SQL 语句,可以共用二级缓存, 二级缓存是跨SqlSession 的。这里需要注意的是,在没有配置的默认情况下, MyBatis 只开启一 级缓存。

My Batis 的缓存模式如图所示:

一级缓存

概述

MyBatis 的一级缓存是SqI Session 级别的缓存, 在操作数据库时需要构造Sq!Session 对象。 每个SqISession 对象中都有一个HashMap 对象用于缓存数据,不同的SqlSession 之间的缓存数 据区域互不影响。 ​ MyBatis的一级缓存作用域是SqlSession 范围的,在参数和SQL 完全一样的情况下,使用 同一个SqlSession 对象调用同一个Mapper 方法,往往只执行一次SQL ,因为MyBatis 会将数据 放在缓存中,下次查询的时候,如果没有声明需要刷新缓存并且缓存没有超时, SqlSession 都 只会取出当前缓存的数据,而不会再次发送SQL 到数据库中。需要注意的是,如果SqlSession 执行了DML 操作( insert 、update 和delete ),并提交到数据库, MyBatis 会清空SqlSession 中 的一级缓存,这样做的目的是为了保证缓存中存储的是最新的信息, 以避免出现脏读现象。 MyBatis 的缓存机制是基于id 进行缓存的, MyBatis 使用HashMap 缓存数据时,使用对象的id 作为key,而对象则作为value 保存。

示例

上述代码中,通过@Resource 注解注入SqlSessionFactoryBean 对象, SqlSessionFactoryBean 对象在applicationContext.xml 配置文件中己配置, 具体代码如下:

通过SqlSessionFactory 工厂获取SqlSession 对象, 通过SqlSession 对象的getMapper()方法 获取AyUserDao 接口对象, 并执行AyUserDao 接口对象的findByld()方法。AyUserDao 和 AyUserMapper.xml 的代码如下所示。

执行测试用例testsessionCach e(),控制台打印相关的信息,具体如图9-2 所示。

由图9 -2 中控制台打印的信息可以看出,第一次查询和第二次查询,查询SQL 的日志只输 出一遍, 这就说明了第二次查询的数据不是从数据库查询出来的,是从一级缓存中获取的。

生命周期

MyBatis 在开启一个Session 会话时, 会创建一个新的SqISession 对象, 每个SqlSession 对 象会创建一个新的Executor 对象, Executor 对象中持有一个新的PerpetualCache 对象: 当会话结 束时, SqISession 对象及其内部的Executor 对象还有PerpetualCache 对象也会一并释放掉,即: ​ (1 )如果SqlSession 调用了close()方法,会释放掉一级缓存PerpetualCache 对象, 一级缓 存不可用。 ​ (2 )如果SqlSession 调用了clearCache(), 会清空PerpetualCache 对象中的数据,但是该对 象仍可使用。 ​ ( 3 ) SqlSession 中执行任何一个DDL 操作( update 、delete 、insert) ,都会清空 PerpetualCache 对象的数据,但是该对象可以继续使用。

二级缓存

概述

二级缓存是Mapper 级别的缓存。使用二级缓存时,多个SqlSession 使用同一个Mapper ( namespace )的SQL 语句操作数据库,得到的数据会存在二级缓存区域,二级缓存同样是使用 HashMap 进行数据存储。二级缓存比一级缓存作用域范围更大,多个SqISession 可以共用二级 缓存,二级缓存是跨SqISession 的。当某个SqISession 类的实例对象执行了增、删、改等操作 时, Mapper 实例会清空二级缓存。MyBatis 默认没有开启二级缓存,需要在配置中开启二级缓 存。开启二级缓存的步骤如下。

首先,在applicationContext.xml 配置文件中添加如下配置:

上面配置信息中最重要的就是指定MyBatis 配置文件的位置:

其次,在src\main\resources 目录下添加配置文件mybatis-config . xml , 具体代码如下:

最后,由于二级缓存是Mapper 级别的,还要在需要开启二级缓存的具体mapper.xml 文件 中开启二级缓存,方法很简单,只需要在mappe川ml 文件中添加一个cache 标签既可,具体代 码如下所示:

cache 标签有很多属性,常用的属性如表9- 1 所示:

示例

AyUserDao 和AyUserMapper.xml 代码如下:

当开启MyBatis 二级缓存后,执行测试用例testSessionCache(), 控制台打印相关的信息, 具体如图9-4 所示。

由图9-4 可知, 第一次查询数据时, 获取连接、编译SQL、加载了数据库中的数据。而第 二次查询数据之前,进行了update 操作,相当于进行commit 操作,也就是说会清空一级缓存来 保证数据的最新状态。但是开启了二级缓存,在第二次查询时,会从二级缓存中获取数据。 这里需要注意的是,如果在select 标签中设置“ userCache = false” 可以禁用当前select 语句 的二级缓存,具体代码如下:

这里简单总结一下二级缓存的特点:

还需要注意的是,使用二级缓存需要特别谨慎,有时候不同的namespace 下的SQL 配置可 能缓存了相同的数据。例如AyUserMapper.xml 中有很多查询缓存了用户数据,其他的 XXXMapper且nl 中有针对用户表进行单表操作, 也缓存了用户数据,如果在AyUserMapper.xml 中做了刷新缓存的操作,在XXXMapper.xml 中的缓存数据仍然有效, 这样在查询数据时可能会 出现脏数据。所以使用MyBatis 的二级缓存时,要根据具体的业务情况,谨慎使用。

###cache-ref共享缓存

MyBatis 并不是整个Application 只有一个Cache 缓存对象,它将缓存划分的更细, 也就是 Mapper 级别的,即每一个Mapper 都可以拥有一个Cache 对象,具体如下:

( 1 ) 为每一个Mapper 分配一个Cache 缓存对象(使用<cache>节点配置) 。 ( 2 )多个Mapper 共用一个Cache 缓存对象(使用<cache-ref>节点配置) 。

如果想让多个Mapper 共用一个Cache ,可以使用<cache-ref namespace="与节点,来指定这 个Mapper 共享哪一个Mapper 的Cache 缓存。具体如图9-5 所示。

mybatis缓存原理

工作原理

如图9- 6 所示, 一个SqlSession 对象中创建一个本地缓存( local cache ),对于每次查询, 都会根据查询条件去一级缓存中查找,如果缓存中存在数据,就直接从缓存中取出,然后返回 给用户:否则,从数据库读取数据,将查询结果存入缓存并返回给用户。

SqlSession 将它的工作交给了Executor 执行器这个角色来完成,负责完成对数据库的各种 操作。当创建一个SqlSession 对象时, MyBatis 会为这个SqlSession 对象创建一个新的Executor 执行器,而缓存信息就被维护在这个Executor 执行器中, MyBatis 将缓存和对缓存相关的操作 封装成了Cache 接口。

如图9 - 7 所示, MyBatis 的二级缓存机制的关键是使用Executor 对象。当开启SqlSession 会 话时, 一个Session 对象使用一个Executor 对象来完成会话操作。如果用户配置了 "cacheEnabled=true”, 那么MyBatis 在为SqlSession 对象创建Executor 对象时,会对Executor 对 象加上一个装饰者: CachingExecutor ,这时SqISession 使用CachingExecutor 对象来完成操作请 求。CachingExecutor 对于查询请求,会先判断该查询请求在二级缓存中是否有缓存结果,如果 有查询结果, 则直接返回缓存结果:如果缓存中没有, 再交给真正的Executor 对象来完成查询 操作,之后CachingExecutor 会将真正Executor 返回的查询结果放置到缓存中,然后再返回给 用户。

mybatis缓存机制(转)的更多相关文章

  1. mybatis缓存机制

    目录 mybatis缓存机制 Executor和缓存 一级缓存 小结 二级缓存 小结 mybatis缓存机制 mybatis支持一.二级缓存来提高查询效率,能够正确的使用缓存的前提是熟悉mybatis ...

  2. 聊聊MyBatis缓存机制【美团-推荐】

    聊聊MyBatis缓存机制 2018年01月19日 作者: 凯伦 文章链接 18778字 38分钟阅读 前言 MyBatis是常见的Java数据库访问层框架.在日常工作中,开发人员多数情况下是使用My ...

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

    <深入理解mybatis原理> MyBatis缓存机制的设计与实现 本文主要讲解MyBatis非常棒的缓存机制的设计原理,给读者们介绍一下MyBatis的缓存机制的轮廓,然后会分别针对缓存 ...

  4. Mybatis缓存机制及mybatis的各个组成部分

    Mybatis 一级缓存: 基于PerpetualCache 的 HashMap本地缓存,其存储作用域为 Session,当 Session flush 或 close 之后,该Session中的所有 ...

  5. 聊聊MyBatis缓存机制

    https://tech.meituan.com/mybatis_cache.html 前言 MyBatis是常见的Java数据库访问层框架.在日常工作中,开发人员多数情况下是使用MyBatis的默认 ...

  6. 【转】MyBatis缓存机制

    转载:https://blog.csdn.net/bjweimengshu/article/details/79988252. 本文转载自公众号 美团技术点评 前言 MyBatis是常见的Java数据 ...

  7. MyBatis 缓存机制(十三)

    什么是缓存 缓存就是内存中的一个对象,用于对数据库查询结果的保存,用于减少与数据库的交互次数从而降低数据库的压力,进而提高响应速度. MyBatis 缓存机制原理 Mybatis 缓存机制原理是将第一 ...

  8. 开源框架是如何使用设计模式的-MyBatis缓存机制之装饰者模式

    写在前面 聊一聊MyBatis是如何使用装饰者模式的,顺便回顾下缓存的相关知识,可以看看右侧目录一览内容概述. 装饰者模式 这里就不了它的概念了,总结下就是套娃.利用组合的方式将装饰器组合进来,增强共 ...

  9. MyBatis缓存机制-二级缓存

    MyBatis二级缓存是基于namespace级别的缓存. 1.MyBatis的缓存机制整体设计以及二级缓存的工作模式 如上图所示,当开一个会话时,一个SqlSession对象会使用一个Executo ...

  10. Mybatis——缓存机制

    MyBatis 包含一个非常强大的查询缓存特性,它可以非常方便地配置和定制.缓存可以极大的提升查询效率. MyBatis系统中默认定义了两级缓存. 一级缓存和二级缓存. 1.默认情况下,只有一级缓存( ...

随机推荐

  1. [Python3] 020 借函数,谈一谈变量的作用域

    目录 1. 概述 2. 分类 3. 变量的作用范围 少废话,上例子 4. 将局部变量提升为全局变量 少废话,上例子 5. 内建函数 globals() 与 locals() 少废话,上例子 6. 邪恶 ...

  2. 如何选择适合自己的Linux版本

    如何选择适合自己的Linux版本: 1.Linux桌面系统,首选Ubuntu; 2.服务器端的Linux系统,首选RHEL或CentOS,这两种中首选CentOS,如果公司有钱,不在乎成本也可以选择R ...

  3. Keyboarding (bfs+预处理+判重优化)

    # #10030. 「一本通 1.4 练习 2」Keyboarding [题目描述] 给定一个 $r$ 行 $c$ 列的在电视上的"虚拟键盘",通过「上,下,左,右,选择」共 $5 ...

  4. P3064 [USACO12DEC]伊斯坦布尔的帮派 (模拟)

    题目传送门 题意: 一片草地,每次可以只可以让一种牛占领,问你怎样安排牛的次序 最后剩下的是1号牛,并且输出其数量 思路: 看到n到100 ,所以可以(n^3)暴力,第一重遍历次序,第二枚举是哪只牛 ...

  5. Thinkphp 获取最大值id值

    有时候项目需要获取数据库最大的id值,比如生成订单,做排序号,那么Thinkphp 如何获取最大值id值. $info=D('Customer')->where('1=1')->order ...

  6. Angular 组件通讯方式

    (一)父子组件 输入/输出属性    关键词  Input,Output,EventEmitter. 父子组件信息信息,分为 (1)子组件向父组件传递 (2)父组件向子组件传递 (二)模版变量与 @V ...

  7. vue项目 PC端点击查看大图

    今天,发现了一款还不错的插件来实现查看大图,成熟度也比较高,支持各种操作 原作品的github地址为 https://github.com/mirari/v-viewer 也有对应的中文文档,使用方法 ...

  8. 了解Greenplum (2)

    一.目的 1. 理解Greenplum中的数据分布策略(random 和 distribution),分析不同分布策略的优劣:2. 理解查询执行中的数据广播和数据重分布,分析在何种情况下选择哪种策略, ...

  9. 编译安装github上的kafka_exporter项目

    本文介绍的kafka_exporter是prometheus监控系统中针对kafka的一款监控插件,要使用这个监控插件,kafka的版本需要满足 0.10.1.0 及以上. 项目的github地址:h ...

  10. nginx windows 代理 80端口 500

    今天准备配置一个nginx 用来代理80端口分别访问.net core 和spring boot 服务器 配置使用的最基本的代理配置 #user nobody; worker_processes 1; ...