【SpringBoot】SpringBoot开启MyBatis缓存+ehcache(一二级缓存和myBatis的差不多,第三方缓存是jar包的不一样)

news2025/1/11 10:57:27

文章目录

    • 第三方缓存
        • 1、导jar包
        • 2、配置文件ehcache.xml
        • 3、@EnableCaching开启缓存
        • 4、application.yml读取配置文件
        • 5、使用缓存(注解@Cacheable)
          • @Cacheable的三个属性:value、key和condition
          • ▶测试:
          • root对象可以用来生成key
          • @CachePut
          • @CacheEvict 清除缓存

第三方缓存

基于springboot的第三方缓存设置

1、导jar包

引入缓存的依赖包,在配置 pom.xml 文件中添加

<!--添加缓存-->
<dependency>
   <groupId>org.springframework.boot</groupId>
   <artifactId>spring-boot-starter-cache</artifactId>
</dependency>
<dependency>
   <groupId>net.sf.ehcache</groupId>
   <artifactId>ehcache</artifactId>
</dependency>

2、配置文件ehcache.xml

创建 .xml 文件,配置缓存内容

<?xml version="1.0" encoding="UTF-8"?>
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd"
         updateCheck="false">
    <!--指定一个文件目录,当EhCache把数据写到硬盘上时,将把数据写到这个文件目录下
             user.home :         用户主目录
             user.dir :          用户当前工作目录
             java.io.tmpdir :    默认临时文件路径
         -->
    <diskStore path="java.io.tmpdir/Tmp_EhCache"/>

	<!--defaultCache:默认缓存策略,当ehcache找不到定义的缓存时,则默认缓存策略-->
    <defaultCache eternal="false" maxElementsInMemory="1000" overflowToDisk="true" diskPersistent="true" timeToIdleSeconds="0" timeToLiveSeconds="600" memoryStoreEvictionPolicy="LRU"/>
    <cache
            name="myCache"
            eternal="false"
            maxElementsInMemory="200"
            overflowToDisk="false"
            diskPersistent="true"
            timeToIdleSeconds="0"
            timeToLiveSeconds="300"
            memoryStoreEvictionPolicy="LRU"/>
</ehcache>

cache中属性的解释:

  • name: 缓存名称
  • eternal: true表示对象永不过期,此时会忽略timeToIdleSeconds和>- timeToLiveSeconds属性,默认为false
  • timeToIdleSeconds: 设定允许对象处于空闲状态的最长时间,以秒为单位。当对象自从最近一次被访问后,如果处于空闲状态的时间超过了timeToIdleSeconds属性值,这个对象就会过期,>- EHCache将把它从缓存中清空。只有当eternal属性为false,该属性才有效。如果该属性值为0,则表示对象可以无限期地处于空闲状态
  • timeToLiveSeconds: 设定对象允许存在于缓存中的最长时间,以秒为单位。当对象自从被存放到缓存中后,如果处于缓存中的时间超过了 timeToLiveSeconds属性值,这个对象就会过期,>- EHCache将把它从缓存中清除。只有当eternal属性为false,该属性才有效。如果该属性值为0,则表示对象可以无限期地存在于缓存中。timeToLiveSeconds必须大于timeToIdleSeconds属性,才有意义
  • maxElementsInMemory: 内存中最大缓存对象数;maxElementsInMemory界限后,会把溢出的对象写到硬盘缓存中。注意:如果缓存的对象要写入到硬盘中的话,则该对象必须实现了Serializable接口才行
  • memoryStoreEvictionPolicy: 当达到maxElementsInMemory限制时,Ehcache将会根据指定的策略去清理内存。可选策略有:LRU(最近最少使用,默认策略)、FIFO(先进先出)、LFU(最少访问次数)
  • maxElementsOnDisk: 硬盘中最大缓存对象数,若是0表示无穷大
  • overflowToDisk: 是否保存到磁盘,当系统宕机时
  • diskPersistent: 是否缓存虚拟机重启期数据,是否持久化磁盘缓存,当这个属性的值为true时,系统在初始化时会在磁盘中查找文件名为cache名称,后缀名为index的文件,这个文件中存放了已经持久化在磁盘中的cache的index,找到后会把cache加载到内存,要想把cache真正持久化到磁盘,写程序时注意执行net.sf.ehcache.Cache.put(Element element)后要调用flush()方法
  • diskSpoolBufferSizeMB: 这个参数设置DiskStore(磁盘缓存)的缓存区大小。默认是30MB。每个Cache都应该有自己的一个缓冲区
  • diskExpiryThreadIntervalSeconds: 磁盘失效线程运行时间间隔,默认为120秒
  • clearOnFlush: 内存数量最大时是否清除

3、@EnableCaching开启缓存

在启动类上开启缓存:

/**
 * 核心注解文件
 */
@SpringBootApplication
@MapperScan("com.hz.springboot01.mapper")
@EnableCaching  //开启缓存
public class SpringBoot01Application {

    public static void main(String[] args) {
        SpringApplication.run(SpringBoot01Application.class, args);
    }
}

4、application.yml读取配置文件

【读取文件的主配置都是 application.xxx ,只是后缀名用的不一样,内容格式不一样而已】

▶在 application.properties 中读取配置文件 ehcache.xml:

#读取缓存配置文件
spring.cache.ehcache.config=classpath:ehcache.xml
#最后记得开启打印sql语句,方便测试
logging.level.com.hz.dao=debug

▶在 application.yml中读取配置文件 ehcache.xml:

# 读取缓存策略文件
spring:
  cache:
    ehcache:
      config: classpath:ehcache.xml  

# mybatis相关配置
mybatis:
  configuration:
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl #SQL控制台输出
    cache-enabled: true #开启mybatis的二级缓存

5、使用缓存(注解@Cacheable)

【之前mybatis的时候,开启缓存在 mapper.xml 文件中开启,现在我们是service 中开启,将service实现中方法的返回值进行缓存】

可以标记在一个方法上,也可以标记在一个类上。
当标记在一个方法上时表示该方法是支持缓存的,当标记在一个类上时则表示该类所有的方法都是支持缓存的。
对于一个支持缓存的方法,Spring会在其被调用后将其返回值缓存起来,以保证下次利用同样的参数来执行该方法时可以直接从缓存中获取结果,而不需要再次执行该方法。

@Cacheable的三个属性:value、key和condition

value:指定哪个缓存
key:如果用相同缓存名称,使用key来区别不同的缓存空间,缓存不共享
condition:指过滤条件,选择谁缓存或者不缓存

    @Cacheable //执行默认策略
    public User find(Integer id) {}
    @Cacheable("myCache1")//Cache是发生在ehcache.xml中myCache1上的
    public User find(Integer id) {
        ....
   }
   @Cacheable({"cache1", "cache2"})//Cache是发生在ehcache.xml中cache1和cache2上的
    public User find(Integer id) {
        .....
   }
   //自定义策略是指我们可以通过Spring的EL表达式来指定我们的key
   //#id指参数id作为key
   @Cacheable(value="myCache1", key="#id")
     public User find(Integer id) {
        ...
   }
   //#p0标识第一个参数作为key
    @Cacheable(value="myCache1", key="#p0")
   public User find(Integer id) {
    .....
   }
    //#user.user_id表示对象user属性user_id作为key
    @Cacheable(value="myCache1", key="#user.user_id")
   public User find(User user) {
    .....
   }
    @Cacheable(value="myCache1", key="#p0.user_id")
   public User find(User user) {
    .....
   }

//表示只有当user的id为偶数时才会进行缓存
@Cacheable(value={"users"}, key="#user.id", condition="#user.id%2==0")
public User find(User user) {
	...
}

▶测试:

前面配置与书写代码完成后,可以使用上次的springboot项目添加缓存进行测试。
【关于分页的缓存:参数过多,所以就直接使用它默认】

@Service
public class ProviderServiceImpl implements ProviderService {

    @Autowired
    private ProviderMapper providerMapper;
    
    @Cacheable("myCache")   //使用缓存
    public List<Provider> providerInfoList(String proName, String startTime, String endTime, Integer page, Integer limit) {
        //计算偏移量
        int pyl = (page-1)*limit;
        return providerMapper.providerInfoList(proName,startTime,endTime,pyl,limit);
    }
}

注意:在实体类实现一下序列化,否则会报一个错误(程序正常运行)
在这里插入图片描述

缓存成功!!!
在这里插入图片描述

root对象可以用来生成key

Spring还为我们提供了一个root对象可以用来生成key

示例描述
root.methodName当前方法名
root.method.name当前方法
root.target当前被调用的对象
root.targetClass当前被调用的对象的class
root.args[0]当前方法参数组成的数组
root.caches[0].name当前被调用的方法使用的Cache
@CachePut

使用@CachePut时我们可以指定的属性跟@Cacheable是一样的

@Cacheable不同的是使用@CachePut标注的方法在执行前不会去检查缓存中是否存在之前执行过的结果,而是每次都会执行该方法,并将执行结果以键值对的形式存入指定的缓存中

@CacheEvict 清除缓存

一般用于增删改上。。。可以指定的属性有value、key、condition、allEntries、beforeInvocation

//如果不指定key,会清除所有"myCache"的缓存
 @CacheEvict(value="myCache",key="#p0.user_id")
    public int updUser(SfUser user) throws Exception {
        return sfUserMapper.updUser(user);
    }
 @CacheEvict(value="users", allEntries=true)
   public void delete(Integer id) {
      System.out.println("delete user by id: " + id);
   }

allEntries:

  • allEntries是boolean类型,表示是否需要清除缓存中的所有元素。
  • 默认为false,表示不需要。当指定了allEntries为true时,Spring Cache将忽略指定的key,删除所有缓存,下次执行的时候会重新加入。
  • 有的时候我们需要Cache一下清除所有的元素,这比一个一个清除元素更有效率

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/28282.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

LeetCode HOT 100 —— 4.寻找两个正序数组的中位数

题目 给定两个大小分别为 m 和 n 的正序&#xff08;从小到大&#xff09;数组 nums1 和 nums2。请你找出并返回这两个正序数组的 中位数 。 算法的时间复杂度应该为 O(log (mn)) 思路 正序数组&#xff0c;立即推—>二分查找 如果本题不要求时间复杂度O&#xff08;log…

光点数据可视化解决方案,助力新型智慧城市打造_光点科技

随着城市化进程的快速发展&#xff0c;智慧城市逐渐从理论理念演变为实践。智慧城市作为一个极其复杂的城市数字化建设和运营系统&#xff0c;涵盖了大量的智能交通、智能物流、智能公园等子系统。对于智慧城市解决方案提供商和集成商来说&#xff0c;其数据可视化产品的统一监…

View的绘制流程

view的绘制流程主要为measure&#xff0c;layout&#xff0c;draw三个阶段 View与window的逻辑结构如图所示&#xff1a; ViewRootImpl&#xff08;替代ViewRoot&#xff09;类&#xff0c;是连接WindowMannager和DecorView的纽带&#xff0c;View的三大流程均是通过ViewRoot完…

godoc安装与go文档查询

前言 最近在用go语言做项目&#xff0c;语法还是很简单的&#xff0c;但是API不熟悉&#xff0c;&#x1f605;&#xff0c;另外也没有类似Java的Spring这样的独秀的IOC和AOP框架&#xff0c;灵活度很大&#xff0c;经常需要查询文档&#xff0c;但是godoc从go1.14.2开始就从内…

【CNN】MobileNet——卷积神经网络中轻量级的经典

前言 MobileNet 系列 是 Andrew G. Howard&#xff08;Google Inc.&#xff09; 等人于 2017 年&#xff08;其实是 2016 年先于 Xception 已经提出&#xff0c;但是直到 2017 年才挂到 arXiv 上&#xff09;在 MobileNets: Efficient Convolutional Neural Networks for Mobi…

干货 | 如何在子查询和联接之间选择

在 联接与子查询&#xff1a;哪个更快&#xff1f;文章我们了解到连接往往比子查询执行得更快。话虽如此&#xff0c;这并不是一条通用准则&#xff0c;因此你可能不希望自动假设联接更可取。正如那篇文章中提到的&#xff0c;如果你需要在查询添加许多联接&#xff0c;数据库服…

分治算法Divide and Conquer

评价 它可以减少运行的时间&#xff0c;很多问题如果暴力求解需要O(n^2)的复杂度&#xff0c;而通过分治可以减少到O&#xff08;nlogn&#xff09; 当与随机化技术相结合时&#xff0c;分治的功能很强大 分治算法的步骤 1.先将大的问题分解为一个个小的子问题 2.对每一个子…

swift内存绑定

swift提供了3种不同的API来绑定/重新绑定指针 assumingMemoryBound(to:)bindMemory(to: capacity:)withMemoryRebound(to: capacity: body:) 绕过编译器检查 - assumingMemoryBound 就是假定内存绑定 func testPointer(_ p: UnsafePointer<Int>) {print(p) } let tup…

Tomcat运行流程、Servlet运行原理以及常用API

文章目录Servlet原理Tomcat 的定位Tomcat 的伪代码Tomcat 初始化流程Tomcat处理请求总结Servlet的核心APIHttpServletHttpServletRequestHttpServletResponseCookie 和 SessionServlet原理 Servlet终究是属于应用层&#xff0c;它是在应用层进行的一系列操作&#xff0c;它的底…

Prometheus Operator 实战 监控 etcd 集群

上节课和大家讲解了 Prometheus Operator 的安装和基本使用方法&#xff0c;这节课给大家介绍如何在 Prometheus Operator 中添加一个自定义的监控项。 除了 Kubernetes 集群中的一些资源对象、节点以及组件需要监控&#xff0c;有的时候我们可能还需要根据实际的业务需求去添…

Java 后端 本地调试-获取微信公众号 openId

Java 后端 本地调试-获取微信公众号 openId申请测试微信公众号内网穿透工具配置公众号获取用户 openId申请测试微信公众号 微信测试公众号 内网穿透工具 netapp 配置公众号 搜索网页账号选项 点击修改&#xff0c;填写内网穿透的域名 获取用户 openId 1 第一步&#xff…

国家高新技术企业的好处

国家高新技术企业的好处&#xff1a;享受税收减免优惠政策&#xff1b;国家科研经费支持和财政拨款&#xff1b;国家级的资质认证硬招牌&#xff1b;提升企业品牌形象&#xff1b;促进企业科技转型&#xff1b;提高企业市场价值&#xff1b;提高企业资本价值&#xff1b;吸引市…

【电脑讲解】电脑如何实现双系统

核心提示&#xff1a;电脑双系统&#xff0c;大家应该不会太陌生&#xff0c;有的网吧就装的是双系统&#xff0c;双系统可以满足不同人群的需要&#xff0c;可以这样说&#xff0c;一个系统可以专门工作使用&#xff0c;另一个可以供玩游戏使用&#xff0c;&#xff08;电脑硬…

[Linux打怪升级之路]-环境变量

前言 作者&#xff1a;小蜗牛向前冲 名言&#xff1a;我可以接受失败&#xff0c;但我不能接受放弃 如果觉的博主的文章还不错的话&#xff0c;还请点赞&#xff0c;收藏&#xff0c;关注&#x1f440;支持博主。如果发现有问题的地方欢迎❀大家在评论区指正。 目录 一、认识环…

一文详解JVM的内存结构

目录 前言 内存结构 程序计数器 虚拟机栈 本地方法栈 堆内存 方法区 内部组成 前言 Java的JVM解决的问题是跨操作系统问题。程序员只需要专注于代码的编写&#xff0c;这些代码能够在不同的操作系统Mac&#xff0c;Linux和Windows运行的前提是JVM。JVM还提供了垃圾回收机制…

Linux终端操作-Xshell和Xftp(家庭版)

目录一&#xff0c;终端操作二&#xff0c;软件安装1&#xff0c;Xshell, Xftp下载2&#xff0c;Xshell安装3&#xff0c;Xftp安装三&#xff0c;使用1&#xff0c;Xshell建立连接2&#xff0c;Xftp上传文件一&#xff0c;终端操作 上一篇博客记录了如何本地安装虚拟机并实现本…

SpringSecurity(十五)---OAuth2的运行机制(上)-OAuth2概念和授权码模式讲解

一、前言 鸽了很久&#xff0c;其实也因为自己确实比较忙&#xff0c;加之自己在造demo的时候也遇到了很多问题&#xff0c;并且网上这方面的解答非常之少&#xff0c;不过也正是因为少&#xff0c;才更加让我想写这样的知识分享&#xff0c;最终&#xff0c;在一篇博客的解答…

中文drupal教程(4)Session会话系统

Session&#xff08;会话&#xff09;在网站中扮演非常重要的角色&#xff0c;储存临时用户数据、登录数据等等都用到了它&#xff0c;Drupal使用到了Symfony的Session组件&#xff0c;该组件非常强大灵活&#xff0c;drupal在此基础上有所改造和扩展&#xff0c;要理解Symfony…

企业微信接口测试实战(一)

本文为在霍格沃兹测试开发学社中学习到的一些技术,写出来分享给大家,希望有志同道合的小伙伴可以一起交流技术,一起进步~ 霍格沃茨启发: 测试开发进阶班>接口自动化测试>企业微信接口测试实战 企业微信接口测试实战 一、准备环境二、脚本实现2.1、 获得access_token2…

防火墙用户管理理论+实验

目录 注&#xff1a;实验需要有安全策略配置、NAT配置基础 一、防火墙用户管理重要知识点 用户管理 访问控制策略 NGFW下一代防火墙 AAA 鉴别方式——认证 用户认证的分类&#xff1a; 上网用户上线流程&#xff1a; 二、用户认证实验&#xff1a; 实验拓扑 先配置防…