草庐IT

Mybatis | Mybatis 一级缓存、二级缓存、三级自定义缓存(Redis)

一介草民丶 2023-05-27 原文

Mybatis 一级缓存、二级缓存、三级自定义缓存Redis实现

一、Mybatis 缓存

Mybatis为什么要有缓存?

       Mybatis 提供缓存当然是为了减少了与SQL数据库的I/O交互, 提升查询效率。
       在Mybatis中存在两种缓存,一级缓存与二级缓存。下面我们说说一级缓存与二缓存

二、一级缓存 SqlSession级别

Mybaits 中一级缓存也就做本地缓存,一级缓存是在会话也就是SqlSession层面实现的,一级缓存的作用范围是在同一个SqlSession中,不同的SqlSession及时查询相同的数据也不会走缓存。

1.如何开启一级缓存

       一级缓存是默认开启的, 需要开启事务。不加事务每次查询都会创建SqlSession.

2.什么时候清除缓存?

  • 当执行增删改方法时会清除缓存。
  • MyBatis全局配置属性localCacheScope配置为Statement时,那么完成一次查询就会清除缓存。
  • 是否配置了flushCache=true属性,如果配置了则会清除一级缓存。

3.什么时候缓存失效?

  • SqlSession 需要开启事务才会生效。否则每次查询都坏创建一个新的SqlSession
  • 不同的SqlSession 对应不同的缓存
  • 同一个 SqlSession 查询条件不同 也会不走缓存。
  • 手动清空缓存,调用SqlSession clearCache()方法。

4.存在的问题

       在同一个SqlSession中 两次相同条件查询中,第一次查询后 然后手动修改表数据或者另一个SqlSession对象修改了数据库或者分布式情况下数据发生了修改 那么第二次查询是直接走缓存,查询结果依旧相同,会存在数据不一致问题。


5.查询 加事务与不加事务

加事务 只创建了一次SqlSession


不加事务 创建了两次

三、二级缓存 Mapper级别

Mybatis 二级缓存是mapper级别的 需要手动开启,他的作用范围更广也就是mapper文件的一个命名空间(namespace)中。

1.开启二级缓存

二级缓存需要手动开启

  1. application.properties 中配置
mybatis-plus.configuration.cache-enabled=true
  1. 在mapper.xml文件中加入下边配置 或者 使用注解SQL开发的话使用@CacheNamespace来开启
<cache/>
或者
<cache-ref namespace="com.xxx.xxx.mapper.UserMapper"/> 
  1. 如果某个sql 不想走缓存的话 需要在像如下操作useCache=false
<select id="findLogs" resultType="map" useCache="false">
  1. 查询返回结果集需要实现序列化接口
  2. 当第一次查询事务提交后, 才会真正的存取二级缓存中

2.缓存清除

  • 当执行增删改方法时会清除缓存。

3.存在的问题

        二级缓存也存在同一级缓存一样的问题,只不过二级缓存的作用域比一级缓存大,

在同一个命名空间(namespace)中 两次相同条件查询中,第一次查询后 然后手动修改表数据或者分布式情况下数据发生了修改 那么再次查询是直接走缓存,查询结果依旧相同,会存在数据不一致问题。

四、三级缓存 第三方自定义缓存(这里用Redis实现)

Mybatis 的一级缓存与二级缓存 只适用于单体项目,在分布式服务或者微服务架构下 都会出现数据不一致问题。所以Mybatis 为我们提供了自定义缓存 我们可以集成很多三方中间件来做缓存 这里就那Redis来说一下。

1.使用 Mybatis整合Redis实现分布式缓存

  1. POM坐标
        <dependency>
            <groupId>org.mybatis.caches</groupId>
            <artifactId>mybatis-redis</artifactId>
            <version>1.0.0-beta2</version>
        </dependency>
  1. application.properties 中配置
# 开启缓存
mybatis-plus.configuration.cache-enabled=true


##redis 配置
redis.host=localhost
redis.port=6379
redis.connectionTimeout=5000
redis.password= 
redis.database=0
  1. mapper.xml 文件中
<cache type="org.mybatis.caches.redis.RedisCache"/>

五、总结

MyBatis 的一级缓存 二级缓存 都不建议使用,它们只适用于单体项目,现在基本都是分布式或者微服务 框架使用的话会存在数据不一致问题。

在我开发这几年,用的都是微服务架构。也没用过Mybatis 的缓存 都是手动在代码中做的redis 缓存。手动在代码中做redis缓存我觉得更灵活。

有关Mybatis | Mybatis 一级缓存、二级缓存、三级自定义缓存(Redis)的更多相关文章

  1. ruby - Facter::Util::Uptime:Module 的未定义方法 get_uptime (NoMethodError) - 2

    我正在尝试设置一个puppet节点,但ruby​​gems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由ruby​​gems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby

  2. ruby-on-rails - Rails 3.2.1 中 ActionMailer 中的未定义方法 'default_content_type=' - 2

    我在我的项目中添加了一个系统来重置用户密码并通过电子邮件将密码发送给他,以防他忘记密码。昨天它运行良好(当我实现它时)。当我今天尝试启动服务器时,出现以下错误。=>BootingWEBrick=>Rails3.2.1applicationstartingindevelopmentonhttp://0.0.0.0:3000=>Callwith-dtodetach=>Ctrl-CtoshutdownserverExiting/Users/vinayshenoy/.rvm/gems/ruby-1.9.3-p0/gems/actionmailer-3.2.1/lib/action_mailer

  3. ruby-on-rails - form_for 中不在模型中的自定义字段 - 2

    我想向我的Controller传递一个参数,它是一个简单的复选框,但我不知道如何在模型的form_for中引入它,这是我的观点:{:id=>'go_finance'}do|f|%>Transferirde:para:Entrada:"input",:placeholder=>"Quantofoiganho?"%>Saída:"output",:placeholder=>"Quantofoigasto?"%>Nota:我想做一个额外的复选框,但我该怎么做,模型中没有一个对象,而是一个要检查的对象,以便在Controller中创建一个ifelse,如果没有检查,请帮助我,非常感谢,谢谢

  4. ruby - 主要 :Object when running build from sublime 的未定义方法 `require_relative' - 2

    我已经从我的命令行中获得了一切,所以我可以运行rubymyfile并且它可以正常工作。但是当我尝试从sublime中运行它时,我得到了undefinedmethod`require_relative'formain:Object有人知道我的sublime设置中缺少什么吗?我正在使用OSX并安装了rvm。 最佳答案 或者,您可以只使用“require”,它应该可以正常工作。我认为“require_relative”仅适用于ruby​​1.9+ 关于ruby-主要:Objectwhenrun

  5. ruby - 在 Ruby 中有条件地定义函数 - 2

    我有一些代码在几个不同的位置之一运行:作为具有调试输出的命令行工具,作为不接受任何输出的更大程序的一部分,以及在Rails环境中。有时我需要根据代码的位置对代码进行细微的更改,我意识到以下样式似乎可行:print"Testingnestedfunctionsdefined\n"CLI=trueifCLIdeftest_printprint"CommandLineVersion\n"endelsedeftest_printprint"ReleaseVersion\n"endendtest_print()这导致:TestingnestedfunctionsdefinedCommandLin

  6. ruby - 定义方法参数的条件 - 2

    我有一个只接受一个参数的方法:defmy_method(number)end如果使用number调用方法,我该如何引发错误??通常,我如何定义方法参数的条件?比如我想在调用的时候报错:my_method(1) 最佳答案 您可以添加guard在函数的开头,如果参数无效则引发异常。例如:defmy_method(number)failArgumentError,"Inputshouldbegreaterthanorequalto2"ifnumbereputse.messageend#=>Inputshouldbegreaterthano

  7. ruby - 如何在 Grape 中定义哈希数组? - 2

    我使用Ember作为我的前端和GrapeAPI来为我的API提供服务。前端发送类似:{"service"=>{"name"=>"Name","duration"=>"30","user"=>nil,"organization"=>"org","category"=>nil,"description"=>"description","disabled"=>true,"color"=>nil,"availabilities"=>[{"day"=>"Saturday","enabled"=>false,"timeSlots"=>[{"startAt"=>"09:00AM","endAt"=>

  8. ruby - 获取模块中定义的所有常量的值 - 2

    我想获取模块中定义的所有常量的值:moduleLettersA='apple'.freezeB='boy'.freezeendconstants给了我常量的名字:Letters.constants(false)#=>[:A,:B]如何获取它们的值的数组,即["apple","boy"]? 最佳答案 为了做到这一点,请使用mapLetters.constants(false).map&Letters.method(:const_get)这将返回["a","b"]第二种方式:Letters.constants(false).map{|c

  9. ruby - 如何在 Ubuntu 中清除 Ruby Phusion Passenger 的缓存? - 2

    我试过重新启动apache,缓存的页面仍然出现,所以一定有一个文件夹在某个地方。我没有“公共(public)/缓存”,那么我还应该查看哪些其他地方?是否有一个URL标志也可以触发此效果? 最佳答案 您需要触摸一个文件才能清除phusion,例如:touch/webapps/mycook/tmp/restart.txt参见docs 关于ruby-如何在Ubuntu中清除RubyPhusionPassenger的缓存?,我们在StackOverflow上找到一个类似的问题:

  10. ruby - 这两个 Ruby 类初始化定义有什么区别? - 2

    我正在阅读一本关于Ruby的书,作者在编写类初始化定义时使用的形式与他在本书前几节中使用的形式略有不同。它看起来像这样:classTicketattr_accessor:venue,:datedefinitialize(venue,date)self.venue=venueself.date=dateendend在本书的前几节中,它的定义如下:classTicketattr_accessor:venue,:datedefinitialize(venue,date)@venue=venue@date=dateendend在第一个示例中使用setter方法与在第二个示例中使用实例变量之间是

随机推荐