Spring Boot 2.x基礎教程:進程內緩存的使用與Cache註解詳解

来源:https://www.cnblogs.com/didispace/archive/2020/07/14/13301480.html
-Advertisement-
Play Games

隨著時間的積累,應用的使用用戶不斷增加,數據規模也越來越大,往往資料庫查詢操作會成為影響用戶使用體驗的瓶頸,此時使用緩存往往是解決這一問題非常好的手段之一。Spring 3開始提供了強大的基於註解的緩存支持,可以通過註解配置方式低侵入的給原有Spring應用增加緩存功能,提高數據訪問性能。 在Spr ...


隨著時間的積累,應用的使用用戶不斷增加,數據規模也越來越大,往往資料庫查詢操作會成為影響用戶使用體驗的瓶頸,此時使用緩存往往是解決這一問題非常好的手段之一。Spring 3開始提供了強大的基於註解的緩存支持,可以通過註解配置方式低侵入的給原有Spring應用增加緩存功能,提高數據訪問性能。

在Spring Boot中對於緩存的支持,提供了一系列的自動化配置,使我們可以非常方便的使用緩存。下麵我們通過一個簡單的例子來展示,我們是如何給一個既有應用增加緩存功能的。

快速入門

下麵我們將使用使用Spring Data JPA訪問MySQL一文的案例為基礎。這個案例中包含了使用Spring Data JPA訪問User數據的操作,利用這個基礎,我們為其添加緩存,來減少對資料庫的IO,以達到訪問加速的作用。如果您還不熟悉如何實現對MySQL的讀寫操作,那麼建議先閱讀前文,完成這個基礎案例的編寫。

先簡單回顧下這個案例的基礎內容:

User實體的定義

@Entity
@Data
@NoArgsConstructor
public class User {

    @Id
    @GeneratedValue
    private Long id;

    private String name;
    private Integer age;

    public User(String name, Integer age) {
        this.name = name;
        this.age = age;
    }
}

User實體的數據訪問實現

public interface UserRepository extends JpaRepository<User, Long> {

    User findByName(String name);

    User findByNameAndAge(String name, Integer age);

    @Query("from User u where u.name=:name")
    User findUser(@Param("name") String name);

}

為了更好的理解緩存,我們先對該工程做一些簡單的改造。

  • application.properties文件中新增spring.jpa.show-sql=true,開啟hibernate對sql語句的列印。如果是1.x版本,使用spring.jpa.properties.hibernate.show_sql=true參數。
  • 修改單元測試類,插入User表一條用戶名為AAA,年齡為10的數據。並通過findByName函數完成兩次查詢,具體代碼如下:
@RunWith(SpringRunner.class)
@SpringBootTest
public class Chapter51ApplicationTests {

    @Autowired
    private UserRepository userRepository;

    @Test
    public void test() throws Exception {
        // 創建1條記錄
        userRepository.save(new User("AAA", 10));

        User u1 = userRepository.findByName("AAA");
        System.out.println("第一次查詢:" + u1.getAge());

        User u2 = userRepository.findByName("AAA");
        System.out.println("第二次查詢:" + u2.getAge());
    }

}

在沒有加入緩存之前,我們可以先執行一下這個案例,可以看到如下的日誌:

Hibernate: select user0_.id as id1_0_, user0_.age as age2_0_, user0_.name as name3_0_ from user user0_ where user0_.name=?
第一次查詢:10
Hibernate: select user0_.id as id1_0_, user0_.age as age2_0_, user0_.name as name3_0_ from user user0_ where user0_.name=?
第二次查詢:10

兩次findByName查詢都執行了兩次SQL,都是對MySQL資料庫的查詢。

引入緩存

第一步:在pom.xml中引入cache依賴,添加如下內容:

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

第二步:在Spring Boot主類中增加@EnableCaching註解開啟緩存功能,如下:

@EnableCaching
@SpringBootApplication
public class Chapter51Application {

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

}

第三步:在數據訪問介面中,增加緩存配置註解,如:

@CacheConfig(cacheNames = "users")
public interface UserRepository extends JpaRepository<User, Long> {

    @Cacheable
    User findByName(String name);

}

第四步:再來執行以下單元測試,可以在控制臺中輸出了下麵的內容

Hibernate: insert into user (age, name, id) values (?, ?, ?)
Hibernate: select user0_.id as id1_0_, user0_.age as age2_0_, user0_.name as name3_0_ from user user0_ where user0_.name=?
第一次查詢:10
第二次查詢:10

到這裡,我們可以看到,在調用第二次findByName函數時,沒有再執行select語句,也就直接減少了一次資料庫的讀取操作。

為了可以更好的觀察,緩存的存儲,我們可以在單元測試中註入CacheManager

@Autowired
private CacheManager cacheManager;

使用debug模式運行單元測試,觀察CacheManager中的緩存集users以及其中的User對象的緩存加深理解。

可以看到,在第一次調用findByName函數之後,CacheManager將這個查詢結果保存了下來,所以在第二次訪問的時候,就能匹配上而不需要再訪問資料庫了。

Cache配置註解詳解

回過頭來我們再來看這裡使用到的兩個註解分別作了什麼事情:

  • @CacheConfig:主要用於配置該類中會用到的一些共用的緩存配置。在這裡@CacheConfig(cacheNames = "users"):配置了該數據訪問對象中返回的內容將存儲於名為users的緩存對象中,我們也可以不使用該註解,直接通過@Cacheable自己配置緩存集的名字來定義。
  • @Cacheable:配置了findByName函數的返回值將被加入緩存。同時在查詢時,會先從緩存中獲取,若不存在才再發起對資料庫的訪問。該註解主要有下麵幾個參數:
    • valuecacheNames:兩個等同的參數(cacheNames為Spring 4新增,作為value的別名),用於指定緩存存儲的集合名。由於Spring 4中新增了@CacheConfig,因此在Spring 3中原本必須有的value屬性,也成為非必需項了
    • key:緩存對象存儲在Map集合中的key值,非必需,預設按照函數的所有參數組合作為key值,若自己配置需使用SpEL表達式,比如:@Cacheable(key = "#p0"):使用函數第一個參數作為緩存的key值,更多關於SpEL表達式的詳細內容可參考官方文檔
    • condition:緩存對象的條件,非必需,也需使用SpEL表達式,只有滿足表達式條件的內容才會被緩存,比如:@Cacheable(key = "#p0", condition = "#p0.length() < 3"),表示只有當第一個參數的長度小於3的時候才會被緩存,若做此配置上面的AAA用戶就不會被緩存,讀者可自行實驗嘗試。
    • unless:另外一個緩存條件參數,非必需,需使用SpEL表達式。它不同於condition參數的地方在於它的判斷時機,該條件是在函數被調用之後才做判斷的,所以它可以通過對result進行判斷。
    • keyGenerator:用於指定key生成器,非必需。若需要指定一個自定義的key生成器,我們需要去實現org.springframework.cache.interceptor.KeyGenerator介面,並使用該參數來指定。需要註意的是:該參數與key是互斥的
    • cacheManager:用於指定使用哪個緩存管理器,非必需。只有當有多個時才需要使用
    • cacheResolver:用於指定使用那個緩存解析器,非必需。需通過org.springframework.cache.interceptor.CacheResolver介面來實現自己的緩存解析器,並用該參數指定。

除了這裡用到的兩個註解之外,還有下麵幾個核心註解:

  • @CachePut:配置於函數上,能夠根據參數定義條件來進行緩存,它與@Cacheable不同的是,它每次都會真是調用函數,所以主要用於數據新增和修改操作上。它的參數與@Cacheable類似,具體功能可參考上面對@Cacheable參數的解析
  • @CacheEvict:配置於函數上,通常用在刪除方法上,用來從緩存中移除相應數據。除了同@Cacheable一樣的參數之外,它還有下麵兩個參數:
    • allEntries:非必需,預設為false。當為true時,會移除所有數據
    • beforeInvocation:非必需,預設為false,會在調用方法之後移除數據。當為true時,會在調用方法之前移除數據。

代碼示例

本文的相關例子可以查看下麵倉庫中的chapter5-1目錄:

如果您覺得本文不錯,歡迎Star支持,您的關註是我堅持的動力!

本文首發:Spring Boot 2.x基礎教程:進程內緩存的使用與Cache註解詳解,轉載請註明出處。
歡迎關註我的公眾號:程式猿DD,獲得獨家整理的學習資源和日常乾貨推送。
如果您對我的其他專題內容感興趣,直達我的個人博客:didispace.com


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • import os import fnmatch ls=os.listdir(r"E:\pdf") #列出文件夾下所有的文件 for 文件名 in ls: if fnmatch.fnmatch(文件名,"[!a-z].pdf"): print(文件名) 1.pdf2.pdf3.pdf4.pdf>>> ...
  • 從接觸 Python 時起,我就覺得 Python 的元組解包(unpacking)挺有意思,非常簡潔好用。 最顯而易見的例子就是多重賦值,即在一條語句中同時給多個變數賦值: >>> x, y = 1, 2 >>> print(x, y) # 結果:1 2 在此例中,賦值操作符“=”號的右側的兩個數 ...
  • 我們為什麼要使用泛型? 在學習集合時,我們都知道集合中是可以存放任意對象的,只要把對象存儲集合後,那麼這時他們都會被提升成Object類型。當我們在取出每一個對象,並且進行相應的操作,這時必須採用向下類型轉換。這個時候就要註意了,很容易出現問題. public class GenericDemo { ...
  • C語言簡介 C 語言是一種通用的高級語言,最初是由丹尼斯·里奇在貝爾實驗室為開發 UNIX 操作系統而設計的。C 語言最開始是於 1972 年在 DEC PDP-11 電腦上被首次實現。 原文鏈接:https://juejin.im/post/5df8c917f265da339772a5d1#he ...
  • 字元串相關的類:String String類:代表字元串,Java 程式中的所有字元串字面值(如 "abc" )都作 為此類的實例實現。 String是一個final類,代表不可變的字元序列 當字元串重新賦值時,需要重新指定記憶體區域,不能使用原有的value進行賦值 當對現有的字元串進行連接操作時, ...
  • 基礎數據類型 在變數的定義中,我們講了每個變數是有類型的,類型在電腦中是用來約束數據的解釋。Go語言和其它電腦語言一樣,提供豐富了豐富的數據類型,我們就來看看到底有哪些類型,同時也可以比較一下它和其它語言的區別。 整型 整型就是用來表示變數是整數的類型。和C類似,Go整型分為兩個大類,無符號和有 ...
  • 作者: zyl910 一、緣由 現在zip類的文件越來越多了,例如jar、docx。 有時我們需批量處理這些文件中的數據,若都是手工操作的話就太麻煩了。於是考慮編程自動處理。 Java提供了ZipInputStream等zip的操作類。但是有些內容比較抽象,沒有代碼範例的話有點難以理解。例如zip中 ...
  • 前言 本文的文字及圖片來源於網路,僅供學習、交流使用,不具有任何商業用途,版權歸原作者所有,如有問題請及時聯繫我們以作處理。 首先,瞭解下什麼是JSON? JSON:JavaScript Object Notation 【JavaScript 對象表示法】 JSON 是一種輕量級的數據交換格式,完全 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...