那曲檬骨新材料有限公司

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

SpringBoot項目中使用緩存的正確方法

jf_ro2CN3Fa ? 來源:芋道源碼 ? 2023-06-13 10:59 ? 次閱讀

前言

緩存可以通過將經常訪問的數據存儲在內存中,減少底層數據源如數據庫的壓力,從而有效提高系統的性能和穩定性。我想大家的項目中或多或少都有使用過,我們項目也不例外,但是最近在review公司的代碼的時候寫的很蠢且low, 大致寫法如下:

publicUsergetById(Stringid){
Useruser=cache.getUser();
if(user!=null){
returnuser;
}
//從數據庫獲取
user=loadFromDB(id);
cahce.put(id,user);
returnuser;
}

其實Spring Boot 提供了強大的緩存抽象,可以輕松地向您的應用程序添加緩存。本文就講講如何使用 Spring 提供的不同緩存注解實現緩存的最佳實踐。

基于 Spring Boot + MyBatis Plus + Vue & Element 實現的后臺管理系統 + 用戶小程序,支持 RBAC 動態權限、多租戶、數據權限、工作流、三方登錄、支付、短信、商城等功能

項目地址:https://github.com/YunaiV/ruoyi-vue-pro

視頻教程:https://doc.iocoder.cn/video/

啟用緩存@EnableCaching

現在大部分項目都是是SpringBoot項目,我們可以在啟動類添加注解@EnableCaching來開啟緩存功能。

@SpringBootApplication
@EnableCaching
publicclassSpringCacheApp{

publicstaticvoidmain(String[]args){
SpringApplication.run(Cache.class,args);
}
}

既然要能使用緩存,就需要有一個緩存管理器Bean,默認情況下,@EnableCaching 將注冊一個ConcurrentMapCacheManager的Bean,不需要單獨的 bean 聲明。ConcurrentMapCacheManager將值存儲在ConcurrentHashMap的實例中,這是緩存機制的最簡單的線程安全實現。

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 實現的后臺管理系統 + 用戶小程序,支持 RBAC 動態權限、多租戶、數據權限、工作流、三方登錄、支付、短信、商城等功能

項目地址:https://github.com/YunaiV/yudao-cloud

視頻教程:https://doc.iocoder.cn/video/

自定義緩存管理器

默認的緩存管理器并不能滿足需求,因為她是存儲在jvm內存中的,那么如何存儲到redis中呢?這時候需要添加自定義的緩存管理器。

添加依賴


org.springframework.boot
spring-boot-starter-data-redis

配置Redis緩存管理器

@Configuration
@EnableCaching
publicclassCacheConfig{

@Bean
publicRedisConnectionFactoryredisConnectionFactory(){
returnnewLettuceConnectionFactory();
}

@Bean
publicCacheManagercacheManager(){
RedisCacheConfigurationredisCacheConfiguration=RedisCacheConfiguration.defaultCacheConfig()
.disableCachingNullValues()
.serializeValuesWith(SerializationPair.fromSerializer(newGenericJackson2JsonRedisSerializer()));

RedisCacheManagerredisCacheManager=RedisCacheManager.builder(redisConnectionFactory())
.cacheDefaults(redisCacheConfiguration)
.build();

returnredisCacheManager;
}
}

現在有了緩存管理器以后,我們如何在業務層面操作緩存呢?

我們可以使用@Cacheable、@CachePut 或@CacheEvict 注解來操作緩存了。

@Cacheable

該注解可以將方法運行的結果進行緩存,在緩存時效內再次調用該方法時不會調用方法本身,而是直接從緩存獲取結果并返回給調用方。

868c1144-098a-11ee-962d-dac502259ad0.png

例子1:緩存數據庫查詢的結果。

@Service
publicclassMyService{

@Autowired
privateMyRepositoryrepository;

@Cacheable(value="myCache",key="#id")
publicMyEntitygetEntityById(Longid){
returnrepository.findById(id).orElse(null);
}
}

在此示例中,@Cacheable 注解用于緩存 getEntityById()方法的結果,該方法根據其 ID 從數據庫中檢索 MyEntity 對象。

但是如果我們更新數據呢?舊數據仍然在緩存中?

@CachePut

然后@CachePut 出來了, 與 @Cacheable 注解不同的是使用 @CachePut 注解標注的方法,在執行前不會去檢查緩存中是否存在之前執行過的結果,而是每次都會執行該方法,并將執行結果以鍵值對的形式寫入指定的緩存中。@CachePut 注解一般用于更新緩存數據,相當于緩存使用的是寫模式中的雙寫模式。

@Service
publicclassMyService{

@Autowired
privateMyRepositoryrepository;

@CachePut(value="myCache",key="#entity.id")
publicvoidsaveEntity(MyEntityentity){
repository.save(entity);
}
}

@CacheEvict

標注了 @CacheEvict 注解的方法在被調用時,會從緩存中移除已存儲的數據。@CacheEvict 注解一般用于刪除緩存數據,相當于緩存使用的是寫模式中的失效模式。

869f9d36-098a-11ee-962d-dac502259ad0.png

@Service
publicclassMyService{

@Autowired
privateMyRepositoryrepository;

@CacheEvict(value="myCache",key="#id")
publicvoiddeleteEntityById(Longid){
repository.deleteById(id);
}
}

@Caching

@Caching 注解用于在一個方法或者類上,同時指定多個 Spring Cache 相關的注解。

86b9efb0-098a-11ee-962d-dac502259ad0.png

例子1:@Caching注解中的evict屬性指定在調用方法 saveEntity 時失效兩個緩存。

@Service
publicclassMyService{

@Autowired
privateMyRepositoryrepository;

@Cacheable(value="myCache",key="#id")
publicMyEntitygetEntityById(Longid){
returnrepository.findById(id).orElse(null);
}

@Caching(evict={
@CacheEvict(value="myCache",key="#entity.id"),
@CacheEvict(value="otherCache",key="#entity.id")
})
publicvoidsaveEntity(MyEntityentity){
repository.save(entity);
}

}

例子2:調用getEntityById方法時,Spring會先檢查結果是否已經緩存在myCache緩存中。如果是,Spring 將返回緩存的結果而不是執行該方法。如果結果尚未緩存,Spring 將執行該方法并將結果緩存在 myCache 緩存中。方法執行后,Spring會根據@CacheEvict注解從otherCache緩存中移除緩存結果。

@Service
publicclassMyService{

@Caching(
cacheable={
@Cacheable(value="myCache",key="#id")
},
evict={
@CacheEvict(value="otherCache",key="#id")
}
)
publicMyEntitygetEntityById(Longid){
returnrepository.findById(id).orElse(null);
}

}

例子3:當調用saveData方法時,Spring會根據@CacheEvict注解先從otherCache緩存中移除數據。然后,Spring 將執行該方法并將結果保存到數據庫或外部 API

方法執行后,Spring 會根據@CachePut注解將結果添加到 myCache、myOtherCache 和 myThirdCache 緩存中。Spring 還將根據@Cacheable注解檢查結果是否已緩存在 myFourthCache 和 myFifthCache 緩存中。如果結果尚未緩存,Spring 會將結果緩存在適當的緩存中。如果結果已經被緩存,Spring 將返回緩存的結果,而不是再次執行該方法。

@Service
publicclassMyService{

@Caching(
put={
@CachePut(value="myCache",key="#result.id"),
@CachePut(value="myOtherCache",key="#result.id"),
@CachePut(value="myThirdCache",key="#result.name")
},
evict={
@CacheEvict(value="otherCache",key="#id")
},
cacheable={
@Cacheable(value="myFourthCache",key="#id"),
@Cacheable(value="myFifthCache",key="#result.id")
}
)
publicMyEntitysaveData(Longid,Stringname){
//CodetosavedatatoadatabaseorexternalAPI
MyEntityentity=newMyEntity(id,name);
returnentity;
}

}

@CacheConfig

通過@CacheConfig 注解,我們可以將一些緩存配置簡化到類級別的一個地方,這樣我們就不必多次聲明相關值:

@CacheConfig(cacheNames={"myCache"})
@Service
publicclassMyService{

@Autowired
privateMyRepositoryrepository;

@Cacheable(key="#id")
publicMyEntitygetEntityById(Longid){
returnrepository.findById(id).orElse(null);
}

@CachePut(key="#entity.id")
publicvoidsaveEntity(MyEntityentity){
repository.save(entity);
}

@CacheEvict(key="#id")
publicvoiddeleteEntityById(Longid){
repository.deleteById(id);
}
}

Condition & Unless

condition作用:指定緩存的條件(滿足什么條件才緩存),可用 SpEL 表達式(如 #id>0,表示當入參 id 大于 0 時才緩存)

unless作用 : 否定緩存,即滿足 unless 指定的條件時,方法的結果不進行緩存,使用 unless 時可以在調用的方法獲取到結果之后再進行判斷(如 #result == null,表示如果結果為 null 時不緩存)

//whenid>10,the@CachePutworks.
@CachePut(key="#entity.id",condition="#entity.id>10")
publicvoidsaveEntity(MyEntityentity){
repository.save(entity);
}


//whenresult!=null,the@CachePutworks.
@CachePut(key="#id",condition="#result==null")
publicvoidsaveEntity1(MyEntityentity){
repository.save(entity);
}

清理全部緩存

通過allEntries、beforeInvocation屬性可以來清除全部緩存數據,不過allEntries是方法調用后清理,beforeInvocation是方法調用前清理。

//方法調用完成之后,清理所有緩存
@CacheEvict(value="myCache",allEntries=true)
publicvoiddelectAll(){
repository.deleteAll();
}

//方法調用之前,清除所有緩存
@CacheEvict(value="myCache",beforeInvocation=true)
publicvoiddelectAll(){
repository.deleteAll();
}

SpEL表達式

Spring Cache注解中頻繁用到SpEL表達式,那么具體如何使用呢?

SpEL 表達式的語法

86cdaee2-098a-11ee-962d-dac502259ad0.png

Spring Cache可用的變量

86e7950a-098a-11ee-962d-dac502259ad0.png

最佳實踐

通過Spring緩存注解可以快速優雅地在我們項目中實現緩存的操作,但是在雙寫模式或者失效模式下,可能會出現緩存數據一致性問題(讀取到臟數據),Spring Cache 暫時沒辦法解決。最后我們再總結下Spring Cache使用的一些最佳實踐。

只緩存經常讀取的數據:緩存可以顯著提高性能,但只緩存經常訪問的數據很重要。很少或從不訪問的緩存數據會占用寶貴的內存資源,從而導致性能問題。

根據應用程序的特定需求選擇合適的緩存提供程序和策略。SpringBoot 支持多種緩存提供程序,包括 Ehcache、Hazelcast 和 Redis。

使用緩存時請注意潛在的線程安全問題。對緩存的并發訪問可能會導致數據不一致或不正確,因此選擇線程安全的緩存提供程序并在必要時使用適當的同步機制非常重要。

避免過度緩存。緩存對于提高性能很有用,但過多的緩存實際上會消耗寶貴的內存資源,從而損害性能。在緩存頻繁使用的數據和允許垃圾收集不常用的數據之間取得平衡很重要。

使用適當的緩存逐出策略。使用緩存時,重要的是定義適當的緩存逐出策略以確保在必要時從緩存中刪除舊的或陳舊的數據。

使用適當的緩存鍵設計。緩存鍵對于每個數據項都應該是唯一的,并且應該考慮可能影響緩存數據的任何相關參數,例如用戶 ID、時間或位置。

常規數據(讀多寫少、即時性與一致性要求不高的數據)完全可以使用 Spring Cache,至于寫模式下緩存數據一致性問題的解決,只要緩存數據有設置過期時間就足夠了。

特殊數據(讀多寫多、即時性與一致性要求非常高的數據),不能使用 Spring Cache,建議考慮特殊的設計(例如使用 Cancal 中間件等)。
責任編輯:彭菁

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 緩存
    +關注

    關注

    1

    文章

    241

    瀏覽量

    26758
  • 代碼
    +關注

    關注

    30

    文章

    4827

    瀏覽量

    69054
  • 數據源
    +關注

    關注

    1

    文章

    63

    瀏覽量

    9718
  • SpringBoot
    +關注

    關注

    0

    文章

    174

    瀏覽量

    201

原文標題:SpringBoot項目中使用緩存的正確姿勢,太優雅了!

文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    如何在XPS項目中使用SmartXplorer

    您好Xilinx社區,我對使用SmartXplorer方法試圖改善設計時間感興趣。我似乎能夠在命令行工具上找到大量信息,并且我了解如何從命令行運行程序等。但是,我似乎無法找到的是如何在XPS項目中使
    發表于 10-17 14:14

    labview項目中的類怎么才能夠遷移到另一個項目使用

    項目中的類怎么才能夠起一道另一個項目中使用,是只能重新在另一個項目中創建么,搞了一天沒有好的方法操作
    發表于 06-03 10:14

    如何在我的項目中使用停止模式?

    你好,我想在我的項目中使用停止模式。有什么例子嗎?我想讓我的外圍模塊在初始化時停止模式。如果用戶將喚醒按鈕,模塊醒來并開始廣告。模塊進入停止模式,再然后preconfiguredtimeout已過期。
    發表于 09-25 14:58

    項目中使用SYSBIOS有好處嗎?

    我找了下ControlSuite發現沒有相應的例程,尤其是驅動部分,另外在項目中使用SYSBIOS有好處嗎?
    發表于 06-01 06:49

    SpringBoot應用啟動運行run方法

    什么時候創建嵌入式的Servlet容器工廠?什么時候獲取嵌入式的Servlet容器并啟動Tomcat;獲取嵌入式的Servlet容器工廠:1)、SpringBoot應用啟動運行run方法2
    發表于 12-20 06:16

    分享一下在項目中使用串口接收數據及處理的方法

    在定時器中斷里需要做哪些事呢?怎樣在項目中使用串口接收數據及處理呢?
    發表于 02-24 06:17

    在ESP-IDF項目中使用BSON有什么想法嗎?

    組件。我需要按塊從相機發送照片,所以我需要將每個塊都以二進制格式放置。在 ESP-IDF 項目中使用 BSON 有什么想法嗎?
    發表于 03-01 06:30

    如何在ESP-IDF項目中使用BSON ?

    組件。我需要按塊從相機發送照片,所以我需要將每個塊都以二進制格式放置。在 ESP-IDF 項目中使用 BSON 有什么想法嗎?
    發表于 04-14 06:59

    使用Method Swizzling遇到的問題和項目中使用的Swizzling方案

    導語:Method Swizzling是Objective-C中運行時中討論較多的內容,本文主要介紹使用Method Swizzling遇到的問題和項目中使用的Swizzling方案。 一
    發表于 09-22 19:35 ?0次下載
    使用Method Swizzling遇到的問題和<b class='flag-5'>項目中使</b>用的Swizzling方案

    如何在SpringBoot項目中實現動態定時任務

    之前寫過文章記錄怎么在SpringBoot項目中簡單使用定時任務,不過由于要借助cron表達式且都提前定義好放在配置文件里,不能在項目運行中動態修改任務執行時間,實在不太靈活。
    的頭像 發表于 09-30 11:16 ?1842次閱讀

    如何在SpringBoot中解決Redis的緩存穿透等問題

    今天給大家介紹一下如何在SpringBoot中解決Redis的緩存穿透、緩存擊穿、緩存雪崩的問題。
    的頭像 發表于 04-28 11:35 ?773次閱讀

    如何正確使用SpringBoot項目中緩存Cache

    緩存可以通過將經常訪問的數據存儲在內存中,減少底層數據源如數據庫的壓力,從而有效提高系統的性能和穩定性。我想大家的項目中或多或少都有使用過,我們項目也不例外,但是最近在review公司的代碼的時候寫的很蠢且low, 大致寫法如下
    的頭像 發表于 05-11 11:01 ?1253次閱讀
    如何<b class='flag-5'>正確</b>使用<b class='flag-5'>SpringBoot</b><b class='flag-5'>項目中</b><b class='flag-5'>緩存</b>Cache

    Springboot項目的集成以及具體使用及配置

    ? 概念 核心組件 API介紹 Springboot集成 具體業務集成 API使用 ? 前言 項目中需要用到工作流引擎來設計部分業務流程,框架選型最終選擇了 Camunda7,關于 Camunda
    的頭像 發表于 07-03 11:18 ?1590次閱讀
    <b class='flag-5'>Springboot</b><b class='flag-5'>項目</b>的集成以及具體使用及配置

    什么是springBoot業務組件化開發?談談SpringBoot業務組件化

    首先,談一談什么是“springBoot業務組件化開發”,最近一直在開發一直面臨這一個問題,就是相同的業務場景場景在一個項目中使用了,又需要再另外一個項目中復用,一遍又一遍的復制代碼,然后想將該業務的代碼在不同的
    的頭像 發表于 07-20 11:30 ?884次閱讀
    什么是<b class='flag-5'>springBoot</b>業務組件化開發?談談<b class='flag-5'>SpringBoot</b>業務組件化

    如何在Rust項目中使用InfluxDB 2.x

    了更好的性能和更好的用戶體驗。Rust語言提供了InfluxDB 2.x的官方客戶端庫,可以方便地在Rust項目中使用InfluxDB 2.x。 本教程將介紹如何在Rust項目中使用InfluxDB
    的頭像 發表于 09-19 16:33 ?720次閱讀
    百家乐官网打法内容介绍| 百家乐娱乐天上人间| 百家乐官网单跳打法| 大发888游戏官网下载| 百家乐金海岸软件| 永利百家乐官网的玩法技巧和规则| 太阳城百家乐官网怎么出千| 怀来县| 永顺县| 灵武市| 娱乐城设计| 大发888投注技巧| 大发888澳88| 大发888bet亚洲| 大发888刮刮了下载| 威尼斯人娱乐下载平台| 有钱人百家乐的玩法技巧和规则| 星港城百家乐娱乐城| 网络百家乐游戏机怎么破解| 太阳城百家乐的破解| 真人百家乐破解软件下载| 百家乐真人秀| 百家乐图淑何看| 环球百家乐官网的玩法技巧和规则| 百家乐官网骗局视频| 免费百家乐官网过滤软件| 百家乐体育直播| 稳赢的百家乐投注方法| 百家乐的规则玩法| 百家乐群bet20| 大发888娱乐城注册送筹码| 大发888大发娱乐城| 大发888娱乐场 17| 澳门博彩 | 真人百家乐官网海立方| 澳门百家乐官网博客| 百家乐官网破解仪| 24山风水发几房| 百家乐五湖四海娱乐城| 威尼斯人娱乐老| 皇冠网219678|