成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

高德地圖一面:聊聊 @Cacheable 注解的原理!

開發
本文我們從源碼角度深度分析了 @Cacheable注解,Spring通過該注解提供了一種簡潔且強大的緩存處理方式。

在 Spring 框架中,@Cacheable注解是什么?它有什么用途?它是如何工作的?這篇文章,我們來聊一聊。@Cacheable注解

一、@Cacheable概述

首先,我們看看@Conditional注解的源碼,截圖如下:

通過源碼可以知道:@Cacheable表示可以緩存調用某個方法(或某個類中的所有方法)的結果的注解,它可以用在類和方法上。更具體地說,@Cacheable用于將方法的結果緩存起來,如果遇到方法并且參數都完全相同的情況,會直接從緩存中獲取結果,而無需執行方法體。

@Cacheable 的工作原理如下:

(1) 第一次調用:調用被 @Cacheable 注解的方法時,Spring 會先檢查緩存中是否存在對應的緩存條目。

  • 如果不存在,方法會被執行,且返回的結果會被存入緩存中。
  • 如果存在,方法不會被執行,直接返回緩存中的結果。

(2) 后續調用:每次調用時,Spring 都會基于方法的參數在緩存中查找對應的條目,存在則直接返回緩存結果,避免了重復計算或訪問數據源。

二、@Cacheable 的使用

下面,我們將通過詳細的示例來介紹 @Cacheable 的使用方法。

1.. 添加依賴

首先,我們需要在項目中添加 Spring 緩存相關的依賴,比如,我們使用 Spring Boot 和 Redis 作為緩存實現,這里以 Maven為例:

<!-- Maven 依賴 -->
<dependencies>
    <!-- Spring Boot Starter Cache -->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-cache</artifactId>
    </dependency>
    <!-- Redis 作為緩存實現 -->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-data-redis</artifactId>
    </dependency>
</dependencies>

2. 啟用緩存

在 Spring Boot應用的啟動類或配置類上添加 @EnableCaching 注解,以啟用緩存支持。示例代碼如下:

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cache.annotation.EnableCaching;

@SpringBootApplication
@EnableCaching // 開啟緩存支持
public class CacheableDemoApplication {
    public static void main(String[] args) {
        SpringApplication.run(CacheableDemoApplication.class, args);
    }
}

3. 使用 @Cacheable 注解

我們可以在需要緩存的方法上添加 @Cacheable 注解,并指定緩存名稱。示例代碼如下:

import org.springframework.cache.annotation.Cacheable;
import org.springframework.stereotype.Service;

@Service
publicclass UserService {

    // 假設這是一個耗時的查詢方法,比如從數據庫中獲取用戶信息
    @Cacheable(cacheNames = "users", key = "#userId")
    public User getUserById(Long userId) {
        simulateSlowService(); // 模擬慢服務
        returnnew User(userId, "User" + userId);
    }

    private void simulateSlowService() {
        try {
            Thread.sleep(3000L); // 模擬3秒延遲
        } catch (InterruptedException e) {
            thrownew IllegalStateException(e);
        }
    }
}

在上述代碼中:

  • cacheNames = "users":指定緩存的名稱為 users??梢岳斫鉃榫彺娴拿臻g。
  • key = "#userId":指定緩存的鍵為方法參數 userId 的值。

4. 測試緩存效果

下面,我們通過調用getUserById方法兩次,第一次會經過延遲,第二次將直接從緩存中獲取來進行測試。示例代碼如下:

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.CommandLineRunner;
import org.springframework.stereotype.Component;

@Component
publicclass CacheTestRunner implements CommandLineRunner {

    @Autowired
    private UserService userService;

    @Override
    public void run(String... args) throws Exception {
        long start = System.currentTimeMillis();
        User user1 = userService.getUserById(1L); // 第一次查詢,耗時
        long end = System.currentTimeMillis();
        System.out.println("First call took: " + (end - start) + "ms");

        start = System.currentTimeMillis();
        User user2 = userService.getUserById(1L); // 第二次查詢,從緩存獲取,快速
        end = System.currentTimeMillis();
        System.out.println("Second call took: " + (end - start) + "ms");
    }
}

運行結果類似于:

First call took: 3005ms
Second call took: 15ms

說明第一次調用執行了方法體并緩存了結果,第二次調用則直接從緩存中獲取。

三、屬性詳解

@Cacheable 注解提供了多個屬性,以便更靈活地控制緩存行為,如下源碼截圖:

下面,我們將對主要屬性進行詳細的說明。

1. cacheNames/value

  • 描述:指定緩存的名稱,可以是一個或多個。
  • 類型:String[]
  • 默認值:無

說明:cacheNames 和 value 是同義屬性,通常使用 cacheNames。指定一個緩存名稱相當于指定一個命名空間,可以在配置緩存管理器時對不同名稱的緩存指定不同的配置。

@Cacheable(cacheNames = "users")
// 或
@Cacheable(value = "users")

2. key

  • 描述:指定緩存的鍵。在 SpEL(Spring Expression Language)表達式中,可以使用方法參數、返回值等。
  • 類型:String
  • 默認值:基于參數的所有方法參數生成的鍵,類似于 SimpleKey 機制。
@Cacheable(cacheNames = "users", key = "#userId")
@Cacheable(cacheNames = "users", key = "#root.methodName + #userId")
  • #userId:使用 userId 參數作為鍵。
  • #a0 或 #p0:使用第一個參數作為鍵。
  • #result.id:使用方法返回值的 id 屬性作為鍵(適用于 key 屬性中的 unless)。

3. keyGenerator

  • 描述:指定自定義的鍵生成器的名稱。與 key 屬性互斥。
  • 類型:String
  • 默認值:"cacheKeyGenerator",即使用配置的默認鍵生成器。
@Cacheable(cacheNames = "users", keyGenerator = "myKeyGenerator")

自定義鍵生成器示例:

@Component("myKeyGenerator")
public class MyKeyGenerator implements KeyGenerator {
    @Override
    public Object generate(Object target, Method method, Object... params) {
        return method.getName() + "_" + Arrays.stream(params).map(Object::toString).collect(Collectors.joining("_"));
    }
}

4. cacheManager

  • 描述:指定用于該注解的緩存管理器的名稱。
  • 類型:String
  • 默認值:使用配置的默認 CacheManager。
@Cacheable(cacheNames = "users", cacheManager = "cacheManager1")

5. cacheResolver

  • 描述:指定緩存解析器,優先級高于 cacheManager 和 cacheNames。
  • 類型:String
  • 默認值:無
@Cacheable(cacheResolver = "myCacheResolver")

6. condition

  • 描述:使用 SpEL 表達式進行條件判斷,決定是否緩存。只有表達式結果為 true 時,才進行緩存。
  • 類型:String
  • 默認值:""(總是緩存)
@Cacheable(cacheNames = "users", condition = "#userId > 10")

上述示例中,只有當 userId 大于 10 時,才緩存結果。

7. unless

  • 描述:與 condition 相反,用來決定是否不緩存。僅當表達式結果為 true 時,不進行緩存。
  • 類型:String
  • 默認值:""(不阻止緩存)
@Cacheable(cacheNames = "users", unless = "#result == null")

上述示例中,只有當方法返回結果為 null 時,不緩存。

8. sync

  • 描述:是否啟用同步緩存。默認值為 false。
  • 類型:boolean
  • 默認值:false

當多個線程同時請求尚未緩存的值時,啟用同步緩存可以防止多線程重復加載緩存。

@Cacheable(cacheNames = "users", sync = true)

9. 綜合示例

@Cacheable(
    cacheNames = "users",
    key = "#userId",
    condition = "#userId > 10",
    unless = "#result == null",
    sync = true
)
public User getUserById(Long userId) {
    // 方法實現
}
  • 緩存名稱為 users
  • 鍵為 userId
  • 僅當 userId 大于 10 時緩存
  • 如果返回結果為 null,則不緩存
  • 啟用同步緩存,防止緩存穿透導致的高并發請求重復加載

四、配置緩存管理器

要使用 @Cacheable,需要配置一個 CacheManager,Spring 提供了多種緩存管理器的實現,如 ConcurrentMapCacheManager(基于本地 ConcurrentHashMap)、RedisCacheManager、EhCacheCacheManager 等。

1. 使用默認的 ConcurrentMapCacheManager

如果沒有特別指定,Spring Boot 會默認使用 ConcurrentMapCacheManager。適用于簡單的開發和測試場景。

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.cache.concurrent.ConcurrentMapCacheManager;

@Configuration
public class CacheConfig {
    
    @Bean
    public ConcurrentMapCacheManager cacheManager() {
        return new ConcurrentMapCacheManager("users", "products");
    }
}

2. 使用 Redis 作為緩存實現

Redis 是一個高性能的內存數據庫,適用于分布式應用的緩存需求。

(1) 配置 Redis 連接

在 application.properties 或 application.yml 中配置 Redis 連接信息。

spring.redis.host=localhost
spring.redis.port=6379

(2) 配置 RedisCacheManager

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.cache.RedisCacheConfiguration;
import org.springframework.data.redis.cache.RedisCacheManager;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.serializer.RedisSerializationContext;
import java.time.Duration;

@Configuration
publicclass RedisCacheConfig {

   @Bean
   public RedisCacheManager cacheManager(RedisConnectionFactory connectionFactory) {
       RedisCacheConfiguration config = RedisCacheConfiguration.defaultCacheConfig()
               .entryTtl(Duration.ofMinutes(60)) // 設置緩存過期時間
               .disableCachingNullValues()
               .serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(new GenericJackson2JsonRedisSerializer()));
       
       return RedisCacheManager.builder(connectionFactory)
               .cacheDefaults(config)
               .build();
   }
}

說明:

  • entryTtl(Duration.ofMinutes(60)):設置緩存的默認過期時間為 60 分鐘。
  • disableCachingNullValues():不緩存 null 值。
  • serializeValuesWith:配置緩存值的序列化方式,建議使用 JSON 序列化,便于調試和跨語言兼容。

3. 多個緩存管理器

你可以配置多個 CacheManager,并通過 cacheManager 屬性在 @Cacheable 注解中指定使用哪個緩存管理器。例如:

@Configuration
public class MultipleCacheConfig {

    @Bean
    public CacheManager cacheManager1(RedisConnectionFactory connectionFactory) {
        // RedisCacheManager 配置
    }

    @Bean
    public CacheManager cacheManager2() {
        // ConcurrentMapCacheManager 配置
    }
}

在 @Cacheable 中指定:

@Cacheable(cacheNames = "users", cacheManager = "cacheManager1")
public User getUserById(Long userId) {
    // 方法實現
}

五、總結

本文,我們從源碼角度深度分析了 @Cacheable注解,Spring通過該注解提供了一種簡潔且強大的緩存處理方式。在實際工作中,我們一定要根據實際情況來選擇合適的緩存策略,另外,在使用緩存的同時,我們也需要注意緩存常見的問題,比如穿透、擊穿和雪崩,并采取相應的解決措施。

責任編輯:趙寧寧 來源: 猿java
相關推薦

2024-09-19 08:51:01

HTTP解密截取

2024-10-15 10:59:18

Spring MVCJava開發

2022-03-30 10:10:17

字節碼??臻g

2025-03-24 09:10:00

Spring注解代碼

2024-10-31 08:50:14

2025-03-24 07:35:00

開發注解Spring

2025-03-25 12:00:00

@Value?Spring開發

2024-09-26 06:48:36

2022-05-11 22:15:51

云計算云平臺

2009-07-30 14:38:36

云計算

2020-09-19 17:46:20

React Hooks開發函數

2011-12-22 20:53:40

Android

2011-12-23 09:43:15

開源開放

2024-05-15 16:41:57

進程IO文件

2017-04-20 17:36:49

云計算

2022-06-21 07:51:06

Redis高可用哨兵進程

2014-12-02 18:23:40

高德地圖

2017-07-21 10:14:41

高德極客地圖高德地圖

2021-06-15 14:33:00

高德百度騰訊

2012-12-19 09:04:29

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国内精品久久久久 | 91在线成人 | 欧美激情区| 激情欧美一区二区三区 | 国产一区二区三区在线视频 | 美女福利网站 | 欧美黄色片| 欧美一区二区激情三区 | 久久国产综合 | 九九免费在线视频 | 亚洲欧美日韩电影 | 亚洲人成人一区二区在线观看 | 欧美一级欧美一级在线播放 | 天天综合日日夜夜 | 最近日韩中文字幕 | 亚洲人成在线观看 | 色综合天天天天做夜夜夜夜做 | 国产福利91精品 | 国产成人在线视频 | 日韩av成人在线 | www.狠狠操 | 日日久| 成人精品一区亚洲午夜久久久 | 久久久精 | 欧美日韩视频在线 | 看片一区 | 91成人在线 | 一区日韩| 日韩欧美精品 | 欧美理论片在线 | 精品二区 | 日韩午夜精品 | 99re在线 | 福利av在线 | 黄色片免费 | 九九亚洲| 综合久久99 | 国产玖玖 | 嫩草影院网址 | 在线观看成人精品 | 欧美成人一级视频 |