当前位置:首页 > 科技  > 软件

SpringBoot中使用Cache提升接口性能详解

来源: 责编: 时间:2023-08-05 11:44:47 5604观看
导读环境:springboot2.3.12.RELEASE + JSR107 + Ehcache + JPASpring 框架从 3.1 开始,对 Spring 应用程序提供了透明式添加缓存的支持。和事务支持一样,抽象缓存允许一致地使用各种缓存解决方案,并对代码的影响最小。从 Spri

环境:springboot2.3.12.RELEASE + JSR107 + Ehcache + JPAmKE28资讯网——每日最新资讯28at.com

Spring 框架从 3.1 开始,对 Spring 应用程序提供了透明式添加缓存的支持。和事务支持一样,抽象缓存允许一致地使用各种缓存解决方案,并对代码的影响最小。从 Spring4.1 版本开始,缓存抽象支持了 JSR-107 注释和更多自定义选项,从而得到了显著的改进。mKE28资讯网——每日最新资讯28at.com

方式1:直接使用spring的注解来实现缓存mKE28资讯网——每日最新资讯28at.com

spring提供了如下注解:mKE28资讯网——每日最新资讯28at.com

@Cacheable 触发缓存机制mKE28资讯网——每日最新资讯28at.com

@CacheEvict 触发缓存回收mKE28资讯网——每日最新资讯28at.com

@CachePut 更新缓存,而不会影响方法的执行mKE28资讯网——每日最新资讯28at.com

@Caching 组合多个缓存操作到一个方法mKE28资讯网——每日最新资讯28at.com

@CacheConfig 类级别共享系诶常见的缓存相关配置mKE28资讯网——每日最新资讯28at.com

<dependency>  <groupId>org.springframework.boot</groupId>  <artifactId>spring-boot-starter-cache</artifactId></dependency>

首先在Service对应的方法是添加注解:mKE28资讯网——每日最新资讯28at.com

@Servicepublic class StorageService {    @Resource  private StorageRepository sr ;    @Cacheable(value = {"cache_storage"}, keyGenerator = "storageKey")  public Storage getStorage(Long id) {    return sr.findById(id).get() ;  }  }
// 这里的keyGenerator是你自定义Key生成的Bean名称@Component("storageKey")public class StorageKeyGenerator implements KeyGenerator {  private static final String  KEY_PREFIX = "storage_" ;    @Override  public Object generate(Object target, Method method, Object... params) {    StringBuilder sb = new StringBuilder() ;    for (Object param : params) {      sb.append(param) ;    }    return KEY_PREFIX + sb.toString() ;  }}

web接口:mKE28资讯网——每日最新资讯28at.com

@RestController@RequestMapping("/storages")public class StorageController {    @Resource  private StorageService storageService ;  @GetMapping("/{id}")  public Object get(@PathVariable("id") Long id) {    return storageService.getStorage(id) ;  }}

测试:mKE28资讯网——每日最新资讯28at.com

第一次访问接口,查看控制台输出了sql语句:mKE28资讯网——每日最新资讯28at.com

图片图片mKE28资讯网——每日最新资讯28at.com

再次访问接口,发现控制台没有再输出任何sql,说明我们的缓存生效了(这里你也可以把这里的注解注释了来看效果)。关于这里的更新缓存,删除缓存就不演示了。接下来完整的演示下JSR107规范中的注解演示:mKE28资讯网——每日最新资讯28at.com

注意在这些注释中我们是可以使用SpEL表达式的:mKE28资讯网——每日最新资讯28at.com

图片图片mKE28资讯网——每日最新资讯28at.com

方式2:使用JSR107和EhcachemKE28资讯网——每日最新资讯28at.com

先来看看Spring与JSR107注解的对照表:mKE28资讯网——每日最新资讯28at.com

图片图片mKE28资讯网——每日最新资讯28at.com

pom.xml中加入依赖:mKE28资讯网——每日最新资讯28at.com

<dependency>  <groupId>org.springframework.boot</groupId>  <artifactId>spring-boot-starter-cache</artifactId></dependency><dependency>  <groupId>mysql</groupId>  <artifactId>mysql-connector-java</artifactId></dependency><dependency>  <groupId>org.ehcache</groupId>  <artifactId>ehcache</artifactId></dependency><dependency>  <groupId>javax.cache</groupId>  <artifactId>cache-api</artifactId></dependency>

Service类:mKE28资讯网——每日最新资讯28at.com

@Servicepublic class StorageService {    @Resource  private StorageRepository sr ;    // 这里的 @CacheValue 说明是要缓存的参数值。  @Transactional  @CachePut(cacheName = "cache_storage", cacheKeyGenerator = JCacheKeyGenerator.class)  public Storage save(@CacheValue Storage storage) {    return sr.saveAndFlush(storage) ;  }  @CacheResult(cacheName = "cache_storage", cacheKeyGenerator = JCacheKeyGenerator.class)  public Storage getStorage(Long id) {    return sr.findById(id).get() ;  }    @Transactional  @CacheRemove(cacheName = "cache_storage", cacheKeyGenerator = JCacheKeyGenerator.class)  public void removeStorage(Long id) {    sr.deleteById(id) ;  }    @Transactional  @CachePut(cacheName = "cache_storage", cacheKeyGenerator = JCacheKeyGenerator.class)  public Storage updateStorage(@CacheValue Storage storage) {    return sr.saveAndFlush(storage) ;  }}// 注意这里的cacheKeyGenerator 必须全部用同一个,// 跟踪了下源码是用的对应的类名key来查找对应的缓存的;一开始我没有用同一个始终不正确。。// 看下图跟踪的代码:

图片图片mKE28资讯网——每日最新资讯28at.com

这里必须要一样哦cacheKeyGeneratormKE28资讯网——每日最新资讯28at.com

缓存Key:JCacheKeyGenerator.javamKE28资讯网——每日最新资讯28at.com

public class JCacheKeyGenerator implements CacheKeyGenerator {  private static final String  KEY_PREFIX = "storage_" ;    @Override  public GeneratedCacheKey generateCacheKey(      CacheKeyInvocationContext<? extends Annotation> cacheKeyInvocationContext) {    CacheInvocationParameter[] params = cacheKeyInvocationContext.getAllParameters() ;    StringBuilder sb = new StringBuilder() ;    for (CacheInvocationParameter param : params) {      if (param.getValue() instanceof Storage) {        Storage s = (Storage) param.getValue() ;        sb.append(s.getId()) ;      } else {        sb.append((Long)param.getValue()) ;      }    }    return new StorageGeneratedCacheKey(KEY_PREFIX + sb.toString()) ;  }    private static class StorageGeneratedCacheKey implements GeneratedCacheKey {    private static final long serialVersionUID = 1L;        private String key ;        public StorageGeneratedCacheKey(String key) {      this.key = key ;    }    @Override    public int hashCode() {      final int prime = 31;      int result = 1;      result = prime * result + ((key == null) ? 0 : key.hashCode());      return result;    }    @Override    public boolean equals(Object obj) {      if (this == obj)        return true;      if (obj == null)        return false;      if (getClass() != obj.getClass())        return false;      StorageGeneratedCacheKey other = (StorageGeneratedCacheKey) obj;      if (key == null) {        if (other.key != null)          return false;        } else if (!key.equals(other.key))          return false;      return true;      }        }}

application.yml配置:mKE28资讯网——每日最新资讯28at.com

spring:  cache:    cacheNames:    - cache_storage    ehcache:      config: classpath:ehcache.xml

ehcache.xmlmKE28资讯网——每日最新资讯28at.com

<?xml versinotallow="1.0" encoding="UTF-8"?><ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"         xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd"         updateCheck="false">  <diskStore path="java.io.tmpdir/Tmp_EhCache"/>     <defaultCache eternal="false" maxElementsInMemory="10000"  overflowToDisk="false" diskPersistent="false" timeToIdleSeconds="1800" timeToLiveSeconds="259200" memoryStoreEvictionPolicy="LRU" />  <cache name="cache_storage" eternal="false" maxElementsInMemory="5000"  overflowToDisk="false" diskPersistent="false" timeToIdleSeconds="1800" timeToLiveSeconds="1800" memoryStoreEvictionPolicy="LRU" /> </ehcache>

测试增删改:mKE28资讯网——每日最新资讯28at.com

先添加个数据:mKE28资讯网——每日最新资讯28at.com

图片图片mKE28资讯网——每日最新资讯28at.com

mKE28资讯网——每日最新资讯28at.com

图片图片mKE28资讯网——每日最新资讯28at.com

成功添加ID为4的信息,Service中的save方法中我们添加了@CachePut注解,接下来我们查询ID为4的信息,看看控制台是否会生成SQL语句。mKE28资讯网——每日最新资讯28at.com

图片图片mKE28资讯网——每日最新资讯28at.com

mKE28资讯网——每日最新资讯28at.com

图片图片mKE28资讯网——每日最新资讯28at.com

控制台没有增加任何的SQL语句,说明save方法加的@CachePut生效了。mKE28资讯网——每日最新资讯28at.com

接着做删除操作:mKE28资讯网——每日最新资讯28at.com

图片图片mKE28资讯网——每日最新资讯28at.com

mKE28资讯网——每日最新资讯28at.com

图片图片mKE28资讯网——每日最新资讯28at.com

mKE28资讯网——每日最新资讯28at.com

ID为4的删除了,接下来再做查询看看:mKE28资讯网——每日最新资讯28at.com

图片图片mKE28资讯网——每日最新资讯28at.com

mKE28资讯网——每日最新资讯28at.com

这说明删除了数据后,缓存也做了删除。这里生成了查询语句。mKE28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-89-0.htmlSpringBoot中使用Cache提升接口性能详解

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: K8S | Service服务发现

下一篇: 一年经验在二线城市面试后端的经验分享

标签:
  • 热门焦点
  • 线程通讯的三种方法!通俗易懂

    线程通信是指多个线程之间通过某种机制进行协调和交互,例如,线程等待和通知机制就是线程通讯的主要手段之一。 在 Java 中,线程等待和通知的实现手段有以下几种方式:Object 类下
  • Rust中的高吞吐量流处理

    作者 | Noz编译 | 王瑞平本篇文章主要介绍了Rust中流处理的概念、方法和优化。作者不仅介绍了流处理的基本概念以及Rust中常用的流处理库,还使用这些库实现了一个流处理程序
  • Flowable工作流引擎的科普与实践

    一.引言当我们在日常工作和业务中需要进行各种审批流程时,可能会面临一系列技术和业务上的挑战。手动处理这些审批流程可能会导致开发成本的增加以及业务复杂度的上升。在这
  • 自动化在DevOps中的力量:简化软件开发和交付

    自动化在DevOps中扮演着重要角色,它提升了DevOps的效能。通过自动化工具和方法,DevOps团队可以实现以下目标:消除手动和重复性任务。简化流程。在整个软件开发生命周期中实现更
  • Python异步IO编程的进程/线程通信实现

    这篇文章再讲3种方式,同时讲4中进程间通信的方式一、 Python 中线程间通信的实现方式共享变量共享变量是多个线程可以共同访问的变量。在Python中,可以使用threading模块中的L
  • 雅柏威士忌多款单品价格大跌,泥煤顶流也不香了?

    来源 | 烈酒商业观察编 | 肖海林今年以来,威士忌市场开始出现了降温迹象,越来越多不断暴涨的网红威士忌也开始悄然回归市场理性。近日,LVMH集团旗下苏格兰威士忌品牌雅柏(Ardbeg
  • ESG的面子与里子

    来源 | 光子星球撰文 | 吴坤谚编辑 | 吴先之三伏大幕拉起,各地高温预警不绝,但处于厄尔尼诺大&ldquo;烤&rdquo;之下的除了众生,还有各大企业发布的ESG报告。ESG是&ldquo;环境保
  • 郭明錤称华为和江淮汽车合作开发问界MPV,定价100万左右、计划明年量产

    8 月 1 日消息,郭明錤今天在 Medium 平台发布博文,称华为正在和江淮汽车合作,开发售价在 100 万元的问界 MPV,预计在 2024 年第 2 季度量产,销量目标为
  • iQOO 11S屏幕细节公布:首发三星2K E6全感屏 安卓最好的直屏手机

    日前iQOO手机官方宣布,新一代电竞旗舰iQOO 11S将会在7月4日19:00正式与大家见面。随着发布时间的日益临近,官方关于该机的预热也更加密集,截至目前已
Top