自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

統(tǒng)一緩存帝國 - 實戰(zhàn) Spring Cache

存儲 存儲軟件
Spring Cache 是 Spring 提供的一整套的緩存解決方案。雖然它本身并沒有提供緩存的實現(xiàn),但是它提供了一整套的接口和代碼規(guī)范、配置、注解等,這樣它就可以整合各種緩存方案了,比如 Redis、Ehcache,我們也就不用關心操作緩存的細節(jié)。

[[408163]]

一、揭開 Spring Cache 的面紗

1.1 現(xiàn)有緩存方案的痛點

試想一種場景:

1.用戶 A 打開 APP,進入到了秒殺商品的詳情頁,那這個商品數(shù)據(jù)我們會先去數(shù)據(jù)庫查詢,然后返回給客戶端。

2.因為有大量用戶短時間內進入到了詳情頁,所以可以把活動列表緩存起來,直接讀緩存就可以了。

3.那下次再查詢商品時,直接去緩存查詢就可以了。如果秒殺商品下架了,緩存的數(shù)據(jù)不會用到了,就把緩存刪掉就可以了。

4.上面幾步看起來也沒啥問題,但是放緩存,刪除緩存這兩步是需要我們去手動寫代碼實現(xiàn)的。有沒有一種方式不用寫操作緩存的代碼?

5.假如現(xiàn)在用的緩存中間件是 Redis,領導說要換成 Ehcache,操作緩存的代碼是不是又得重新擼一遍?

總結下上面場景的痛點:

需要手寫操作緩存代碼,如添加緩存、更新緩存、刪除緩存。

切換緩存組件并不容易,或者說沒有對緩存層進行抽象封裝,依賴具體的緩存中間件。

哪有沒有一種方案可以幫助解決上面的兩個痛點呢?

這就是今天要介紹的 Spring Cache。

1.2 Spring Cache 介紹

Spring Cache 是 Spring 提供的一整套的緩存解決方案。雖然它本身并沒有提供緩存的實現(xiàn),但是它提供了一整套的接口和代碼規(guī)范、配置、注解等,這樣它就可以整合各種緩存方案了,比如 Redis、Ehcache,我們也就不用關心操作緩存的細節(jié)。

Spring 3.1 開始定義了 org.springframework.cache.Cache 和 org.springframework.cache.CacheManager 接口來統(tǒng)一不同的緩存技術,并支持使用注解來簡化我們開發(fā)。

Cache 接口它包含了緩存的各種操作方式,同時還提供了各種xxxCache緩存的實現(xiàn),比如 RedisCache 針對Redis,EhCacheCache 針對 EhCache,ConcurrentMapCache 針對 ConCurrentMap,具體有哪幾種,后面實戰(zhàn)中會介紹。

1.3 Spring Cache 有什么功效

每次調用某方法,而此方法又是帶有緩存功能時,Spring 框架就會檢查指定參數(shù)的那個方法是否已經(jīng)被調用過,如果之前調用過,就從緩存中取之前調用的結果;如果沒有調用過,則再調用一次這個方法,并緩存結果,然后再返回結果,那下次調用這個方法時,就可以直接從緩存中獲取結果了。

1.4 Spring Cache 的原理是什么?

Spring Cache 主要是作用在類上或者方法上,對類中的方法的返回結果進行緩存。那么如何對方法增強,來實現(xiàn)緩存的功能?

學過 Spring 的同學,肯定能一下子就反應過來,就是用 AOP(面向切面編程)。

面向切面編程可以簡單地理解為在類上或者方法前加一些說明,就是我們常說的注解。

Spring Cache 的注解會幫忙在方法上創(chuàng)建一個切面(aspect),并觸發(fā)緩存注解的切點(poinitcut),聽起來太繞了,簡單點說就是:Spring Cache 的注解會幫忙在調用方法之后,去緩存方法調用的最終結果,或者在方法調用之前拿緩存中的結果,或者刪除緩存中的結果,這些讀、寫、刪緩存的臟活都交給 Spring Cache 來做了,是不是很爽,再也不用自己去寫緩存操作的邏輯了。

1.5 緩存注解

Spring 提供了四個注解來聲明緩存規(guī)則。@Cacheable,@CachePut,@CacheEvict,@Caching。

大家先有個概念,后面我們再來看怎么使用這些緩存注解。

二、使用緩存

2.1 引入 Spring Cache 依賴

在 pom 文件中引入 spring cache 依賴,如下所示:

  1. <dependency> 
  2.     <groupId>org.springframework.boot</groupId> 
  3.     <artifactId>spring-boot-starter-cache</artifactId> 
  4. </dependency> 

2.2 配置使用哪種緩存

Spring Cache 支持很多緩存中間件作為框架中的緩存,總共有 9 種選擇:

  • caffeine:Caffeine 是一種高性能的緩存庫,基于 Google Guava。
  • couchbase:CouchBase是一款非關系型JSON文檔數(shù)據(jù)庫。
  • generic:由泛型機制和 static 組合實現(xiàn)的泛型緩存機制。
  • hazelcast:一個高度可擴展的數(shù)據(jù)分發(fā)和集群平臺,可用于實現(xiàn)分布式數(shù)據(jù)存儲、數(shù)據(jù)緩存。
  • infinispan:分布式的集群緩存系統(tǒng)。
  • jcache:JCache 作為緩存。它是 JSR107 規(guī)范中提到的緩存規(guī)范。
  • none:沒有緩存。
  • redis:用 Redis 作為緩存
  • simple:用內存作為緩存。

mark

我們還是用最熟悉的 Redis 作為緩存吧。配置 Redis 作為緩存也很簡單,在配置文件 application.properties 中設置緩存的類型為 Redis 就可以了, 如下所示:

當然,別忘了還要在 pom 文件中 引入 Redis 的依賴,不然用不了 Redis。

  1. <dependency> 
  2.     <groupId>org.springframework.boot</groupId> 
  3.     <artifactId>spring-boot-starter-data-redis</artifactId> 
  4. </dependency> 

2.3 測試緩存

那基礎的配置已經(jīng)做好了,現(xiàn)在就是看怎么使用 Spring Cache 了。

(1)啟動類上添加 @EnableCaching注解。本文案例就是在 啟動類 PassjavaQuestionApplication 添加 @EnableCaching注解。

(2)指定某方法開啟緩存功能。在方法上添加 @Cacheable 緩存注解就可以了。

@Cacheable 注解中,可以添加四種參數(shù):value,key,condition,unless。首先我們來看下 value 參數(shù)。

下面的代碼出于演示作用,用了最簡單的邏輯,test 方法直接返回一個數(shù)字,連數(shù)據(jù)庫查詢都沒有做。不過沒關系,我們主要驗證 Spring Cache 是否對方法的結果進行了緩存。

  1. @RequestMapping("/test"
  2. @Cacheable({"hot"}) 
  3. public int test() { 
  4.     return 222; 

大家注意,@Cacheable 注解中小括號里面還含有大括號,大括號里面還有 “hot” 字符串,這個 hot 字符串你可以把它當作一個緩存的名字,然后將 test 方法返回的結果存到 hot 緩存中。我們也可以用 value="hot" 的方式。

第一次調用 test 方法前,既沒有 hot 緩存,更沒有 test 的結果緩存。

調用 test 方法后,Redis 中就創(chuàng)建出了 hot 緩存了,然后緩存了一個 key,如下圖所示:

第二次調用 test 方法時,就從緩存 hot 中將 test 方法緩存的結果 222 取出來了,為了驗證沒有執(zhí)行 test 中的方法,大家可以在 test 方法中打下 log 或者斷點。最后的驗證結果肯定是沒有走 test 方法的,而是直接從緩存中獲取的。

那我們再來測試一個方法,方法名改為 test2,且請求路徑也改為 test2 了。

  1. @RequestMapping("/test2"
  2. @Cacheable({"hot"}) 
  3. public int test2() { 
  4.     return 456; 

大家覺得這兩個方法的結果都會緩存嗎?還是只會緩存第一個被調用的方法?

經(jīng)過測試,執(zhí)行第一個 test 方法后,再執(zhí)行 test2 方法,緩存結果一直是 222 不會變。因為他們的 key 都是 默認的 SimpleKey[],所以兩個方法對應的緩存的 key 都叫這個,所以得到的緩存值是一樣的。

(3)加上數(shù)據(jù)庫查詢的測試。

有的同學可能覺得上面的測試太簡單了,test 方法里面啥都沒做,還緩存啥呢,完全沒必要啊。沒關系,大家的顧慮是對的,我們來加上數(shù)據(jù)庫查詢,安排~

先說下場景:前端需要查詢某個題目的詳情,正常邏輯是查詢數(shù)據(jù)庫后返回結果。假定這個查詢操作非常頻繁,我們需要將題目詳情進行緩存。我們先看看常規(guī) Redis 緩存方案:

先從 Redis 緩存中查看緩存中是否有該題目,如果緩存中有,則返回緩存中的題目;如果沒有,就從數(shù)據(jù)庫中查。查詢出題目后,就用 Redis 存起來,然后返回。這里就要寫操作 Redis 的代碼了:查詢 Redis 緩存、更新 Redis 緩存。

  1. // 查詢緩存,假定該題目詳情緩存的 key=question1 
  2. redisTemplate.opsForValue().get("question1");  
  3. // 更新緩存 
  4. redisTemplate.opsForValue().set("question1", questionEntity); 

那如果用 Spring Cache 注解的話,上面兩行代碼可以直接干掉了。如下所示,加一個 @Cacheable 注解搞定。

  1. @Cacheable({"question""hot"}) 
  2. public QuestionEntity info(Long id) { 
  3.     return getById(id); // 查詢數(shù)據(jù)庫操作 

其中 question 和 hot 是緩存的名字,我們可以將結果放到不同的緩存中。

結論:

  • 如果沒有指定請求參數(shù),則緩存生成的 key name,是默認自動生成的,叫做 SimpleKey[]。
  • 如果指定了請求參數(shù),則緩存的 key name 就是請求參數(shù),比如上面 info 方法,key 等于我傳入的 id = 1。
  • 緩存中 key 對應的 value 默認使用 JDK 序列化后的數(shù)據(jù)。
  • value 的過期時間為 -1,表示永不過期。

2.4 自定義配置類

上面保存的緩存數(shù)據(jù)都是默認設置,我們也可以自己定義配置,如下所示,在配置文件 application.properties 添加如下配置:

  1. # 使用 Redis 作為緩存組件 
  2. spring.cache.type=redis 
  3. # 緩存過期時間為 3600s 
  4. spring.cache.redis.time-to-live=3600000 
  5. # 緩存的鍵的名字前綴 
  6. spring.cache.redis.key-prefix=passjava_ 
  7. # 是否使用緩存前綴 
  8. spring.cache.redis.use-key-prefix=true 
  9. # 是否緩存控制,防止緩存穿透 
  10. spring.cache.redis.cache-null-values=true 

然后需要加一個配置類:MyCacheConfig??梢栽谖业拈_源項目 passjava 獲取完整源碼。

  1. RedisCacheConfiguration redisCacheConfiguration(CacheProperties cacheProperties) { 

2.5 自定義 key

然后我們可以指定 key 的值,可以在 @Cacheable 注解里面加上 key 的值 #root.method.name。這是一種特有的表達式,稱作 SpEL 表達式,這里代表用方法名作為緩存 key 的名字。

  1. @Cacheable(value = {"hot"}, key = "#root.method.name"

接下來就是見證奇跡的時刻,調用 test 方法和 test2 方法,發(fā)現(xiàn)有兩個不同的 key,一個是 passjava_test1,另外一個 passjava_test2,它們的 key 就是前綴 passjava_ + 方法名 組成。

SpEL 表達式還有很多其它規(guī)則,如下所示:

可以根據(jù)項目需要選擇合適的表達式來自定義 key。

2.6 自定義條件

除了設置緩存條目的 key,我們還可以自定義條件來決定是否將緩存功能關閉。這里就要用到@Cacheable 另外兩個屬性:condition 和 unless,它倆的格式還是用 SpEL 表達式。對應的四個屬性總結如下:

代碼示例如下:

  1. @Cacheable(value = "hot", unless = "#result.message.containss('NoCache')"

當放回的結果 message 字段包含有 NoCache 就不會進行緩存。

2.7 更新注解

@CachePut 也是用來更新緩存,和 @Cacheable 非常相似,不同點是 @CachePut 注解的方法始終都會執(zhí)行,返回值也會也會放到緩存中。通常用在保存的方法上。

保存成功后,可以將 key 設置保存實例的 id。這個怎么做呢?

之前我們說過 key 可以通過 SpEL 表達式來指定,這里就可以搭配 #result.id 來實現(xiàn)。

這里還是用個例子來說明用法:創(chuàng)建題目的方法,返回題目實例,其中包含有題目 id。

  1. @RequestMapping("/create"
  2. @CachePut(value = "hot"key = "#result.id"
  3. public QuestionEntity create(@Valid @RequestBody QuestionEntity question){ 
  4.     return IQuestionService.createQuestion(question); 

保存的 id 是自增的,值為 123,所以緩存中的 key = passjava_123。

2.8 刪除緩存注解

@CacheEvict 注解的方法在調用時不會在緩存中添加任何東西,但是會從從緩存中移除之前的緩存結果。

示例代碼如下:

  1. @RequestMapping("/remove/{id}"
  2. @CacheEvict(value = "hot"
  3. public R remove(@PathVariable("id") Long id){ 
  4.     IQuestionService.removeById(id); 
  5.     return R.ok(); 

刪除條目的 key 與傳遞進來的 id 相同。我測試的時候傳的 id = 123,經(jīng)過前綴passjava_組裝后就是 passjava_123,所以將之前緩存的 passjava_123 刪除了。重復執(zhí)行也不會報錯。

注意:@CacheEvict 和 @Cacheable、@CachePut 不同,它能夠應用在返回值為 void 的方法上。

@CacheEvict 還有些屬性可供使用,總結如下:

三、 總結

本文通過傳統(tǒng)使用緩存的方式的痛點引出 Spring 框架中的 Cache 組件。然后詳細介紹了 Spring Cache 組件的用法:

  • 五大注解。@Cacheable、@CachePut、@CacheEvict、@Caching,、@CacheConfig。
  • 如何自定義緩存條目的 key。
  • 如何自定義 Cache 配置。
  • 如何自定義緩存的條件。

當然 Spring Cache 并不是萬能的,緩存一致性問題依舊存在,下一篇,我們再細聊緩存的一致性問題。

本文轉載自微信公眾號「悟空聊架構」,可以通過以下二維碼關注。轉載本文請聯(lián)系悟空聊架構公眾號。

 

責任編輯:武曉燕 來源: 悟空聊架構
相關推薦

2023-05-05 18:38:33

多級緩存Caffeine開發(fā)

2023-10-30 07:56:46

Spring緩存

2017-05-18 14:14:25

過濾器Spring ClouZuul

2017-07-31 15:47:50

Zuul統(tǒng)一處理

2017-05-19 15:13:05

過濾器Spring ClouZuul

2025-01-23 08:24:36

2009-09-22 10:50:04

Hibernate c

2023-11-09 08:01:41

Spring緩存注解

2022-03-31 09:13:49

Cache緩存高并發(fā)

2021-07-06 08:59:38

頁緩存PageCache

2018-07-14 21:59:57

緩存數(shù)據(jù)庫數(shù)據(jù)

2021-04-21 21:09:36

緩存系統(tǒng)高可用

2021-12-26 08:08:00

緩存Cache開發(fā)

2025-03-26 03:25:00

SpringGuavaCaffeine

2010-09-26 08:46:06

HTML 5Cache Manif

2014-11-04 10:34:27

JavaCache

2022-03-09 18:54:30

HTTP緩存協(xié)議cache

2018-11-30 15:17:38

CPUCache緩存行

2023-11-30 07:00:56

SpringBoot處理器

2025-02-13 00:34:22

Spring對象系統(tǒng)
點贊
收藏

51CTO技術棧公眾號