国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

高德地圖一面:聊聊 @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開發

2025-03-24 09:10:00

Spring注解代碼

2022-03-30 10:10:17

字節碼??臻g

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

云計算云平臺

2020-09-19 17:46:20

React Hooks開發函數

2009-07-30 14:38:36

云計算

2011-12-23 09:43:15

開源開放

2011-12-22 20:53:40

Android

2024-05-15 16:41:57

進程IO文件

2017-04-20 17:36:49

云計算

2022-06-21 07:51:06

Redis高可用哨兵進程

2025-08-11 05:00:00

2017-07-21 10:14:41

高德極客地圖高德地圖

2014-12-02 18:23:40

高德地圖

2025-07-15 03:00:00

點贊
收藏

51CTO技術棧公眾號

97精品视频在线看| 欧美激情啊啊啊| 成人在线观看免费| 91福利小视频| 成人国产免费电影| 日韩禁在线播放| 四虎影视国产精品| 欧美激情小视频| 91综合久久一区二区| 精品久久久久久一区| 久久av老司机精品网站导航| 日韩精品在线观看av| 国产精品进线69影院| 看电影就来5566av视频在线播放| 欧美一区二区三区在线看| 中文字幕不卡三区视频| 欧美激情精品久久久久久大尺度| 精品久久不卡| 丝袜美腿玉足3d专区一区| 99在线热播精品免费| 在线观看日韩av| 欧美黑人一区| 国产精品a久久久久久| 午夜在线精品| 国产精品又粗又长| 精品国产91乱高清在线观看 | 国产精品视频久久久久| 亚洲免费高清| 久久九九国产视频| 欧美午夜不卡在线观看免费| av在线日韩| 国产欧美韩国高清| 国产自产高清不卡| 三上悠亚在线免费观看| 精品性高朝久久久久久久| 你微笑时很美电视剧整集高清不卡| 精品国产乱码久久久久久88av | 亚洲国产精品久久久久| 偷拍亚洲精品| 色女人综合av| 亚洲精品一卡二卡| 国产精品电影| 国产精品一区二区在线| 成人黄页在线观看| 三区四区电影在线观看| 亚州精品天堂中文字幕| 日韩高清一区二区| 动漫黄在线观看| 久久夜色精品国产欧美乱| 亚洲色图16p| 国产丝袜一区二区三区免费视频| 国产成人ay| 无码粉嫩虎白一线天在线观看 | 日韩小视频在线观看专区| 日本欧美高清| 50度灰在线观看| 在线观看国产日韩| 国产精品2023| 国产成人在线小视频| 欧美性三三影院| 久久影院资源站| 免费看欧美黑人毛片| 欧美日韩美少妇| 九一亚洲精品| 国产欧美在线一区| 亚洲欧美一区二区三区久久 | 欧美在线观看视频在线| 亚洲精品国模| 免费午夜视频在线观看| 日韩精品极品视频| 精品9999| 欧美zozo| 91精品久久久久久久久久久| 中文字幕在线一区二区三区| 亚洲综合伊人| 超碰成人免费在线| 精品久久久久久最新网址| 亚洲破处大片| 簧片在线观看| 秋霞午夜一区二区| 国产精品久久一级| 国产在线一区不卡| 免费一级特黄特色毛片久久看| 精品一区电影国产| 精品一区二区三区在线播放视频| bt在线麻豆视频| 欧美福利精品| 日韩视频一区二区在线观看| 在线成人国产| 9i精品一二三区| 99热国产免费| 在线观看精品一区| 精品成人久久| 黄色在线免费| 亚洲成人网上| 亚洲国产中文字幕久久网| 欧美亚洲三区| 污视频免费在线观看| 欧美日韩日本网| 欧美一级在线免费| 丝袜亚洲另类欧美| 欧美性爽视频| av不卡在线免费观看| 亚洲精品黄网在线观看| 国产福利一区二区三区在线视频| 国产精品久久久久av电视剧| 狠狠干 狠狠操| 久久精品国产69国产精品亚洲| 97久久精品人人做人人爽| 欧美成人精品午夜一区二区| 男女免费网站| 成人精品在线观看| 欧美三区免费完整视频在线观看| 欧美三级特黄| 四虎影院观看视频在线观看| 亚洲欧洲日本国产| 久久久久亚洲| 91在线高清| 亚洲乱码一区二区三区三上悠亚 | 农村寡妇一区二区三区| 精品盗摄一区二区三区| 国产一区不卡精品| 久久亚洲精精品中文字幕| 初尝黑人巨炮波多野结衣电影| 国产美女直播视频一区| 欧美高清视频不卡网| 黑人巨大精品欧美黑白配亚洲| 欧美风情在线视频| 日本免费一二区| 蜜桃网站成人| 日韩亚洲第一页| 亚洲一区二区四区蜜桃| 国产欧美日韩一区二区三区在线| 亚洲黄色中文字幕| 国产videos| 激情小说网站亚洲综合网 | 欧美经典一区| 久草电影在线| 日韩精彩视频| xxxx性欧美| 五月激情综合色| 蜜桃传媒麻豆第一区在线观看| 999久久久国产999久久久| 亚洲一区二区三区成人| 精品91一区二区三区| 午夜精品久久久久久久99热| 色综合久久久久综合体| 国内一区二区视频| 黄色不卡一区| 牛牛精品视频在线| 亚洲xxxx2d动漫1| 日韩一区在线播放| 性欧美长视频| 欧美性生活一级片| av在线free| 黄网免费视频| 在线观看日韩片| 国产va免费精品高清在线| 日韩欧美国产高清| 国产精品不卡一区| 日本在线不卡一区| 曰本一区二区三区视频| 免费看男女www网站入口在线| 黑人巨大精品欧美一区二区奶水| 日韩av在线一区二区三区| 55夜色66夜色国产精品视频| 日韩久久精品一区| 一区二区三区四区精品在线视频| 美女视频一区二区三区| 亚洲欧美成人vr| 美女日韩欧美| 精品亚洲综合| 亚洲久久中文字幕| 黄色一级视频播放| aa成人免费视频| 久久久久久成人精品| 欧美变态凌虐bdsm| 亚洲一区影音先锋| 国产成人av在线影院| 一区在线免费观看| 台湾佬综合网| 欧美美女被草| 欧美黑人猛交的在线视频| 超碰在线12| 97成人在线观看视频| 亚洲一区影院| 国产精品一区二区你懂得| 97在线日本国产| 亚洲人成伊人成综合网久久久 | 亚洲第一搞黄网站| 91伊人久久大香线蕉| 久久人人精品| 在线成人超碰| 最新亚洲精品| 亚洲精品一区二区在线播放∴| 欧美日韩在线资源| 欧美亚洲日本| 中文字幕97| 黄网视频在线观看| 国产一级特黄a大片免费|