對spring的@Cacheable緩存的使用理解
1 什么是緩存
第一個問題,首先要搞明白什么是緩存,緩存的意義是什么。
對于普通業(yè)務(wù),如果要查詢一個數(shù)據(jù),一般直接select數(shù)據(jù)庫進(jìn)行查找。但是在高流量的情況下,直接查找數(shù)據(jù)庫就會成為性能的瓶頸。因為數(shù)據(jù)庫查找的流程是先要從磁盤拿到數(shù)據(jù),再刷新到內(nèi)存,再返回數(shù)據(jù)。磁盤相比于內(nèi)存來說,速度是很慢的,為了提升性能,就出現(xiàn)了基于內(nèi)存的緩存。
這種基于內(nèi)存的緩存,由于無法跟磁盤頻繁進(jìn)行存儲,所以無法保證數(shù)據(jù)的完整性,隨時有可能丟失,所以架構(gòu)一般使用數(shù)據(jù)庫加緩存的方式,數(shù)據(jù)庫用來持久化數(shù)據(jù),緩存用來處理大流量。
2 本地緩存和集中式緩存
緩存按照存儲方式可以分為這本地緩存和集中式緩存。
本地緩存顧名思義就是存儲在本地上,例如靜態(tài)變量就可以說是一種本地緩存,存儲在了JVM中,或者說自己本地搭建的項目用的redis也算是本地緩存,因為緩存和應(yīng)用都在一臺機(jī)器上。
本地緩存效率很高,直接讀取內(nèi)存,沒有網(wǎng)絡(luò)延遲,但是可用性很低,因為出現(xiàn)單點故障的話,數(shù)據(jù)庫和系統(tǒng)都會宕機(jī)。
對于大型項目來說,都會有集中式緩存,例如redis集群。緩存和應(yīng)用服務(wù)器是分離的,服務(wù)器需要通過網(wǎng)絡(luò)請求從緩存獲取數(shù)據(jù),一般應(yīng)用服務(wù)器也會采取集群的方式,這樣可以保證高可用,數(shù)據(jù)不易丟失,而且也能保證各個服務(wù)器的緩存數(shù)據(jù)一致。
對于分布式應(yīng)用來說,本地緩存還會出現(xiàn)緩存不一致的問題,因為每個服務(wù)器的本地緩存都是獨立的。
3 本地緩存的優(yōu)點
剛才說了這么多本地緩存的缺點,那為什么還要用呢?
因為如果都放在集中式緩存中,網(wǎng)絡(luò)延遲會成為性能的瓶頸。因為不在本地內(nèi)存,讀取的時間需要加上網(wǎng)絡(luò)通信的時間。所以在對性能要求更大或者緩存內(nèi)容不需要持久化、不需要一致性的情況下,本地緩存更適合。
所以一般的大型項目都采用本地緩存和集中式緩存混合使用的方式。
4 Spring對于緩存的支持
終于說到正題,本地緩存可以通過spring更簡單的管理和使用。
springboot和springmvc都支持緩存,其中CacheManager是Spring提供的緩存接口。
4.1 spring支持的CacheManager
CacheManager | 描述 |
SimpleCacheManager | 使用簡單的 Collection 來存儲緩存,主要用于測試 |
ConcurrentMapCacheManager | 使用 ConcurrentMap 來存儲緩存 |
NoOpCacheManager | 僅測試用途,不會實際緩存數(shù)據(jù) |
EhCacheCacheManager | 使用 EhCache 作為緩存技術(shù) |
GuavaCacheManager | 使用 Google Guava 的 GuavaCache 作為緩存技術(shù) |
HazelcastCacheManager | 使用 Hazelcast 作為緩存技術(shù) |
JCacheCacheManager | 支持 JCache(JSR-107) 標(biāo)準(zhǔn)的實現(xiàn)作為緩存技術(shù),如 ApacheCommonsJCS |
RedisCacheManager | 使用 Redis 作為緩存技術(shù) |
看著非常多,實際上正常用的只有ConcurrentMapCacheManager,EhCacheCacheManager,GuavaCacheManager(一般使用redis,我們需要更靈活的對redis鍵值進(jìn)行操作,所以不用RedisCacheManager),我們重點去講一下這個GuavaCacheManager。
4.2 GuavaCache
Guava是谷歌開源的Java庫,其中的代表就有這個緩存。
GuavaCache的原理大概是LRU+ConcurrentHashMap,加載在JVM的本地緩存
4.3 引入依賴
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-cache</artifactId>
</dependency>
<dependency>
<groupId>com.google.guava</groupId>
<artifactId>guava</artifactId>
<version>18.0</version>
</dependency>
//有可能需要這個
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context-support</artifactId>
<version>4.2.5.RELEASE</version>
</dependency>4.4 創(chuàng)建配置類
@EnableCaching
@Configuration
public class GuavaCacheConfig {
@Bean
public CacheManager cacheManager() {
GuavaCacheManager cacheManager = new GuavaCacheManager();
cacheManager.setCacheBuilder(
CacheBuilder.newBuilder().
expireAfterWrite(3, TimeUnit.MINUTES));
return cacheManager;
}
}@EnableCaching用來開啟注解功能,這里設(shè)置的失效時間是3分鐘。
Guava Cache 除了代碼中提到的設(shè)置緩存過期時間的策略外,還有其他的策略。下面是 Guava Cache 設(shè)置緩存過期時間的策略:
- expireAfterAccess: 當(dāng)緩存項在指定的時間段內(nèi)沒有被讀或?qū)懢蜁换厥铡?/li>
- expireAfterWrite:當(dāng)緩存項在指定的時間段內(nèi)沒有更新就會被回收,如果我們認(rèn)為緩存數(shù)據(jù)在一段時間后數(shù)據(jù)不再可用,那么可以使用該種策略。
- refreshAfterWrite:當(dāng)緩存項上一次更新操作之后的多久會被刷新。
4.5 緩存注解
標(biāo)題終于出現(xiàn)了
注解 | 解釋 |
@Cacheable | 在方法執(zhí)行前 Spring 先查看緩存中是否有數(shù)據(jù),若有,則直接返回緩存數(shù)據(jù);若無數(shù)據(jù),調(diào)用方法將方法返回值放入緩存中 |
@CachePut | 無論怎樣,都會將方法的返回值放到緩存中。 |
@CacheEvict | 將一條或多條數(shù)據(jù)從緩存中刪除 |
@Caching | 可以通過 @Caching 注解組合多個注解策略在一個方法上 |
我這里就主要解釋下@Cacheable的用法,因為這個比較常見(其他的我也沒用過)
4.6 @Cacheable的用法
常用參數(shù)有
參數(shù) | 解釋 |
value | 名字1 |
key | 名字2 |
condition | 條件判斷,condition="#id>1"表示id大于1的才緩存 |
unless | 條件判斷 ,unless="#id==1"表示id等于1的不緩存 |
#代表的是EL表達(dá)式
這里的key和value和我們以為的緩存鍵值對是不一樣的
value+key 只是我們緩存鍵的名字,真正的值是方法的返回值。
舉一個例子
@Cacheable(value = "olympic_match_new_action",key = "'get_relate_news_'+#rsc")
public List<MatchNewsVO> getRelateNews(String rsc){
....
}一般value取service名,key取方法名,取名按照數(shù)據(jù)庫的下劃線方式。后面那個#rsc指的是傳進(jìn)來的參數(shù),這些都是鍵。返回的List就是緩存的值。
5 @Cacheable失效的原因
在配置正常的情況下,本人親歷的失效原因就是一個類的方法調(diào)用了帶有緩存的方法,結(jié)果緩存失效。
我使用service的A方法,想調(diào)用這個service的緩存B方法,這樣是不行的。
原因是@Cacheable是由AOP代理實現(xiàn),生成了帶有緩存的代理類。其他類想調(diào)用這個類的緩存方法時,會去調(diào)用這個代理類的方法,實現(xiàn)緩存功能。但是類內(nèi)部調(diào)用這個方法,就不會去調(diào)用代理類的方法,導(dǎo)致緩存失效
6 總結(jié)
網(wǎng)上關(guān)于spring本地緩存的文章很少很雜,我稍微總結(jié)了一下發(fā)了出來,有自己的理解也有網(wǎng)上的摘抄。
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
- SpringBoot中緩存@Cacheable出錯的問題解決
- Spring Boot 緩存注解@Cacheable、@CachePut、@CacheEvict超詳細(xì)實戰(zhàn)
- SpringBoot緩存抽象@Cacheable與緩存管理器配置方法
- Spring @Cacheable自定義緩存過期時間的實現(xiàn)示例
- spring @Cacheable擴(kuò)展實現(xiàn)緩存自動過期時間及自動刷新功能
- Springboot使用@Cacheable注解實現(xiàn)數(shù)據(jù)緩存
- Spring Cache @Cacheable 緩存在部分Service中不生效的解決辦法
相關(guān)文章
java實現(xiàn)docker鏡像上傳到harbor倉庫的方式
這篇文章主要介紹了java實現(xiàn)docker鏡像上傳到harbor倉庫的方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2025-06-06
在IntelliJ IDEA 搭建springmvc項目配置debug的教程詳解
這篇文章主要介紹了在IntelliJ IDEA 搭建springmvc項目配置debug的教程詳解,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-09-09
Java讀取properties文件連接數(shù)據(jù)庫的方法示例
這篇文章主要介紹了Java讀取properties文件連接數(shù)據(jù)庫的方法示例,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-04-04
基于Map的computeIfAbsent的使用場景和使用方式
這篇文章主要介紹了基于Map的computeIfAbsent的使用場景和使用方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-09-09
SpringBoot?基于?MongoTemplate?的工具類過程詳解
MongoDB是一個高性能,開源,無模式的文檔型數(shù)據(jù)庫,是當(dāng)前NoSql數(shù)據(jù)庫中比較熱門的一種,這篇文章主要介紹了SpringBoot基于MongoTemplate的工具類,需要的朋友可以參考下2023-09-09

