成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

您的位置:首頁技術文章
文章詳情頁

詳解SpringBoot的三種緩存技術(Spring Cache、Layering Cache 框架、Alibaba JetCache 框架)

瀏覽:112日期:2023-04-16 09:46:07

引言

​前兩天在寫一個實時數據處理的項目,項目要求是 1s 要處理掉 1k 的數據,這時候顯然光靠查數據庫是不行的,技術選型的時候老大跟我提了一下使用 Layering-Cache 這個開源項目來做緩存框架。

​之間問了一下身邊的小伙伴,似乎對這塊了解不多。一般也就用用 Redis 來緩存,應該是很少用多級緩存框架來專門性的管理緩存吧。

​趁著這個機會,我多了解了一些關于 SpringBoot 中緩存的相關技術,于是有了這篇文章!

在項目性能需求比較高時,就不能單單依賴數據庫訪問來獲取數據了,必須引入緩存技術。

常用的有本地緩存、Redis 緩存。

本地緩存:也就是內存,速度快,缺點是不能持久化,一旦項目關閉,數據就會丟失。而且不能滿足分布式系統的應用場景(比如數據不一致的問題)。 Redis 緩存:也就是利用數據庫等,最常見的就是 Redis。Redis 的訪問速度同樣很快,可以設置過期時間、設置持久化方法。缺點是會受到網絡和并發訪問的影響。

本節介紹三種緩存技術:Spring Cache、Layering Cache 框架、Alibaba JetCache 框架。示例使用的 SpringBoot 版本是 2.1.3.RELEASE。非 SpringBoot 項目請參考文章中給出的文檔地址。

項目源碼地址:https://github.com/laolunsi/spring-boot-examples

一、Spring Cache

Spring Cache 是 Spring 自帶的緩存方案,使用簡單,既可以使用本地緩存,也可以使用 Redis

CacheType 包括:

GENERIC, JCACHE, EHCACHE, HAZELCAST, INFINISPAN, COUCHBASE, REDIS, CAFFEINE, SIMPLE, NONE

Spring Cache 的使用很簡單,引入 即可,我這里使用創建的是一個 web 項目,引入的 `spring-boot-starter-web` 包含了 。

這里利用 Redis 做緩存,再引入 spring-boot-starter-data-redis 依賴:

<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId></dependency><!--Redis--><dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId></dependency>

在配置類 or Application 類上添加 @EnableCaching 注解以啟動緩存功能。

配置文件很簡潔(功能也比較少):

server: port: 8081 servlet: context-path: /apispring: cache: type: redis redis: host: 127.0.0.1 port: 6379 database: 1

下面我們編寫一個對 User 進行增刪改查的 Controller,實現對 User 的 save/delete/findAll 三個操作。為演示方便,DAO 層不接入數據庫,而是使用 HashMap 來直接模擬數據庫操作。

我們直接看 service 層的接口實現:

@Servicepublic class UserServiceImpl implements UserService { @Autowired private UserDAO userDAO; @Override @Cacheable(value = 'user', key = '#userId') public User findById(Integer userId) { return userDAO.findById(userId); } @Override @CachePut(value = 'user', key = '#user.id', condition = '#user.id != null') public User save(User user) { user.setUpdateTime(new Date()); userDAO.save(user); return userDAO.findById(user.getId()); } @Override @CacheEvict(value = 'user', key = '#userId') public boolean deleteById(Integer userId) { return userDAO.deleteById(userId); } @Override public List<User> findAll() { return userDAO.findAll(); }}

我們可以看到使用了 @Cacheable、@CachePut、@CacheEvict 注解。

Cacheable:啟用緩存,首先從緩存中查找數據,如果存在,則從緩存讀取數據;如果不存在,則執行方法,并將方法返回值添加到緩存 @CachePut:更新緩存,如果 condition 計算結果為 true,則將方法返回值添加到緩存中 @CacheEvict:刪除緩存,根據 value 與 key 字段計算緩存地址,將緩存數據刪除

測試發現默認的對象存到 Redis 后是 binary 類型,我們可以通過修改 RedisCacheConfiguration 中的序列化規則去調整。比如:

@Configurationpublic class RedisConfig extends CachingConfigurerSupport { @Bean public RedisCacheConfiguration redisCacheConfiguration(){ Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<>(Object.class); RedisCacheConfiguration configuration = RedisCacheConfiguration.defaultCacheConfig(); configuration = configuration.serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(jackson2JsonRedisSerializer)).entryTtl(Duration.ofDays(30)); return configuration; }}

Spring Cache 的功能比較單一,例如不能實現緩存刷新、二級緩存等功能。下面介紹一個開源項目:Layering-Cache,該項目實現了緩存刷新、二級緩存(一級內存、二級 Redis)。同時較容易擴展實現為自己的緩存框架。

二、Layering Cache 框架

文檔:https://github.com/xiaolyuh/layering-cache/wiki/文檔

引入依賴:

<dependency> <groupId>com.github.xiaolyuh</groupId> <artifactId>layering-cache-starter</artifactId> <version>2.0.7</version> </dependency>

配置文件不需要做什么修改。啟動類依然加上 @EnableCaching 注解。

然后需要配置一下 RedisTemplate:

@EnableCaching@Configurationpublic class RedisConfig extends CachingConfigurerSupport { @Bean public RedisTemplate redisTemplate(RedisConnectionFactory redisConnectionFactory) { return createRedisTemplate(redisConnectionFactory); } public RedisTemplate createRedisTemplate(RedisConnectionFactory redisConnectionFactory) { RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>(); redisTemplate.setConnectionFactory(redisConnectionFactory); // 使用Jackson2JsonRedisSerialize 替換默認序列化 Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class); ObjectMapper objectMapper = new ObjectMapper(); objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY); objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL); jackson2JsonRedisSerializer.setObjectMapper(objectMapper); // 設置value的序列化規則和 key的序列化規則 redisTemplate.setKeySerializer(new StringRedisSerializer()); redisTemplate.setValueSerializer(jackson2JsonRedisSerializer); //Map redisTemplate.setHashKeySerializer(new StringRedisSerializer()); redisTemplate.setHashValueSerializer(jackson2JsonRedisSerializer); redisTemplate.afterPropertiesSet(); return redisTemplate; }}

下面我們使用 layering 包中的 @Cacheable @CachePut @CatchEvict 三個注解來替換 Spring Cache 的默認注解。

@Servicepublic class UserServiceImpl implements UserService { @Autowired private UserDAO userDAO; @Override //@Cacheable(value = 'user', key = '#userId') @Cacheable(value = 'user', key = '#userId', firstCache = @FirstCache(expireTime = 5, timeUnit = TimeUnit.MINUTES), secondaryCache = @SecondaryCache(expireTime = 10, preloadTime = 3, forceRefresh = true, isAllowNullValue = true, timeUnit = TimeUnit.MINUTES)) public User findById(Integer userId) { return userDAO.findById(userId); } @Override //@CachePut(value = 'user', key = '#user.id', condition = '#user.id != null') @CachePut(value = 'user', key = '#user.id', firstCache = @FirstCache(expireTime = 5, timeUnit = TimeUnit.MINUTES), secondaryCache = @SecondaryCache(expireTime = 10, preloadTime = 3, forceRefresh = true, isAllowNullValue = true, timeUnit = TimeUnit.MINUTES)) public User save(User user) { user.setUpdateTime(new Date()); userDAO.save(user); return userDAO.findById(user.getId()); } @Override //@CacheEvict(value = 'user', key = '#userId') @CacheEvict(value = 'user', key = '#userId') public boolean deleteById(Integer userId) { return userDAO.deleteById(userId); } @Override public List<User> findAll() { return userDAO.findAll(); }}

三、Alibaba JetCache 框架

文檔:https://github.com/alibaba/jetcache/wiki/Home_CN

JetCache是一個基于Java的緩存系統封裝,提供統一的API和注解來簡化緩存的使用。 JetCache提供了比SpringCache更加強大的注解,可以原生的支持TTL、兩級緩存、分布式自動刷新,還提供了Cache接口用于手工緩存操作。 當前有四個實現,RedisCache、TairCache(此部分未在github開源)、CaffeineCache(in memory)和一個簡易的LinkedHashMapCache(in memory),要添加新的實現也是非常簡單的。

全部特性:

通過統一的API訪問Cache系統 通過注解實現聲明式的方法緩存,支持TTL和兩級緩存 通過注解創建并配置Cache實例 針對所有Cache實例和方法緩存的自動統計 Key的生成策略和Value的序列化策略是可以配置的 分布式緩存自動刷新,分布式鎖 (2.2+) 異步Cache API (2.2+,使用Redis的lettuce客戶端時) Spring Boot支持

SpringBoot 項目中,引入如下依賴:

<dependency> <groupId>com.alicp.jetcache</groupId> <artifactId>jetcache-starter-redis</artifactId> <version>2.5.14</version></dependency>

配置:

server: port: 8083 servlet: context-path: /apijetcache: statIntervalMinutes: 15 areaInCacheName: false local: default: type: caffeine keyConvertor: fastjson remote: default: expireAfterWriteInMillis: 86400000 # 全局,默認超時時間,單位毫秒,這里設置了 24 小時 type: redis keyConvertor: fastjson valueEncoder: java #jsonValueEncoder #java valueDecoder: java #jsonValueDecoder poolConfig: minIdle: 5 maxIdle: 20 maxTotal: 50 host: ${redis.host} port: ${redis.port} database: 1redis: host: 127.0.0.1 port: 6379

Application.class

@EnableMethodCache(basePackages = 'com.example.springcachealibaba')@EnableCreateCacheAnnotation@SpringBootApplicationpublic class SpringCacheAlibabaApplication { public static void main(String[] args) { SpringApplication.run(SpringCacheAlibabaApplication.class, args); }}

字如其意,@EnableMethodCache 用于注解開啟方法上的緩存功能,@EnableCreateCacheAnnotation 用于注解開啟 @CreateCache 來引入 Cache Bean 的功能。兩套可以同時啟用。

這里以上面對 User 的增刪改查功能為例:

3.1 通過 @CreateCache 創建 Cache 實例

@Servicepublic class UserServiceImpl implements UserService { // 下面的示例為使用 @CreateCache 注解創建 Cache 對象來緩存數據的示例 @CreateCache(name = 'user:', expire = 5, timeUnit = TimeUnit.MINUTES) private Cache<Integer, User> userCache; @Autowired private UserDAO userDAO; @Override public User findById(Integer userId) { User user = userCache.get(userId); if (user == null || user.getId() == null) { user = userDAO.findById(userId); } return user; } @Override public User save(User user) { user.setUpdateTime(new Date()); userDAO.save(user); user = userDAO.findById(user.getId()); // cache userCache.put(user.getId(), user); return user; } @Override public boolean deleteById(Integer userId) { userCache.remove(userId); return userDAO.deleteById(userId); } @Override public List<User> findAll() { return userDAO.findAll(); }}

3.2 通過注解實現方法緩存

@Servicepublic class UserServiceImpl implements UserService { // 下面為使用 AOP 來緩存數據的示例 @Autowired private UserDAO userDAO; @Autowired private UserService userService; @Override @Cached(name = 'user:', key = '#userId', expire = 1000) //@Cached( name = 'user:', key = '#userId', serialPolicy = 'bean:jsonPolicy') public User findById(Integer userId) { System.out.println('userId: ' + userId); return userDAO.findById(userId); } @Override @CacheUpdate(name = 'user:', key = '#user.id', value = '#user') public User save(User user) { user.setUpdateTime(new Date()); boolean res = userDAO.save(user); if (res) { return userService.findById(user.getId()); } return null; } @Override @CacheInvalidate(name = 'user:', key = '#userId') public boolean deleteById(Integer userId) { return userDAO.deleteById(userId); } @Override public List<User> findAll() { return userDAO.findAll(); }}

這里用到了三個注解:@Cached/@CacheUpdate/@CacheInvalidate,分別對應著 Spring Cache 中的 @Cacheable/@CachePut/@CacheEvict

具體含義可以參考:https://github.com/alibaba/jetcache/wiki/MethodCache_CN

3.3 自定義序列化器

默認的 value 存儲格式是 binary 的,JetCache 提供的 Redis key 和 value 的序列化器僅有 java 和 kryo 兩種。可以通過自定義序列化器來實現自己想要的序列化方式,比如 json。

JetCache 開發者提出:

jetcache老版本中是有三個序列化器的:java、kryo、fastjson。 但是fastjson做序列化兼容性不是特別好,并且某次升級以后單元測試就無法通過了,怕大家用了以后覺得有坑,就把它廢棄了。 現在默認的序列化器是性能最差,但是兼容性最好,大家也最熟悉的java序列化器。

參考原倉庫中 FAQ 中的建議,可以通過兩種方式來定義自己的序列化器。

3.3.1 實現 SerialPolicy 接口

第一種方式是定義一個 SerialPolicy 的實現類,然后將其注冊成一個 bean,然后在 @Cached 中的 serialPolicy 屬性中指明 bean:name

比如:

import com.alibaba.fastjson.JSONObject;import com.alicp.jetcache.CacheValueHolder;import com.alicp.jetcache.anno.SerialPolicy;import java.util.function.Function;public class JsonSerialPolicy implements SerialPolicy { @Override public Function<Object, byte[]> encoder() { return o -> { if (o != null) { CacheValueHolder cacheValueHolder = (CacheValueHolder) o; Object realObj = cacheValueHolder.getValue(); String objClassName = realObj.getClass().getName(); // 為防止出現 Value 無法強轉成指定類型對象的異常,這里生成一個 JsonCacheObject 對象,保存目標對象的類型(比如 User) JsonCacheObject jsonCacheObject = new JsonCacheObject(objClassName, realObj); cacheValueHolder.setValue(jsonCacheObject); return JSONObject.toJSONString(cacheValueHolder).getBytes(); } return new byte[0]; }; } @Override public Function<byte[], Object> decoder() { return bytes -> { if (bytes != null) { String str = new String(bytes); CacheValueHolder cacheValueHolder = JSONObject.parseObject(str, CacheValueHolder.class); JSONObject jsonObject = JSONObject.parseObject(str); // 首先要解析出 JsonCacheObject,然后獲取到其中的 realObj 及其類型 JSONObject jsonOfMy = jsonObject.getJSONObject('value'); if (jsonOfMy != null) { JSONObject realObjOfJson = jsonOfMy.getJSONObject('realObj'); String className = jsonOfMy.getString('className'); try { Object realObj = realObjOfJson.toJavaObject(Class.forName(className)); cacheValueHolder.setValue(realObj); } catch (ClassNotFoundException e) { e.printStackTrace(); } } return cacheValueHolder; } return null; }; }}

注意,在 JetCache 的源碼中,我們看到實際被緩存的對象的 CacheValueHolder,這個對象包括了一個泛型字段 V,這個 V 就是實際被緩存的數據。為了將 JSON 字符串和 CacheValueHolder(包括了泛型字段 V )進行互相轉換,我在轉換過程中使用 CacheValueHolder 和一個自定義的 JsonCacheObject 類,其代碼如下:

public class JsonCacheObject<V> { private String className; private V realObj; public JsonCacheObject() { } public JsonCacheObject(String className, V realObj) { this.className = className; this.realObj = realObj; } // ignore get and set methods}

然后定義一個配置類:

@Configurationpublic class JetCacheConfig { @Bean(name = 'jsonPolicy') public JsonSerializerPolicy jsonSerializerPolicy() { return new JsonSerializerPolicy(); }}

使用很簡單,比如:

@Cached( name = 'user:', key = '#userId', serialPolicy = 'bean:jsonPolicy')

這種序列化方法是局部的,只能對單個緩存生效。

下面介紹如何全局序列化方法。

3.3.2 全局配置 SpringConfigProvider

JetCache 默認提供了兩種序列化規則:KRYO 和 JAVA (不區分大小寫)。

這里在上面的 JSONSerialPolicy 的基礎上,定義一個新的 SpringConfigProvider:

@Configurationpublic class JetCacheConfig { @Bean public SpringConfigProvider springConfigProvider() { return new SpringConfigProvider() { @Override public Function<byte[], Object> parseValueDecoder(String valueDecoder) { if (valueDecoder.equalsIgnoreCase('myJson')) { return new JsonSerialPolicy().decoder(); } return super.parseValueDecoder(valueDecoder); } @Override public Function<Object, byte[]> parseValueEncoder(String valueEncoder) { if (valueEncoder.equalsIgnoreCase('myJson')) { return new JsonSerialPolicy().encoder(); } return super.parseValueEncoder(valueEncoder); } }; }}

這里使用了類型 myJson 作為新序列化類型的名稱,這樣我們就可以在配置文件的 jetcache.xxx.valueEncoder 和 jetcache.xxx.valueDecoder 這兩個配置項上設置值 myJson/java/kryo 三者之一了。

關于 Java 中緩存框架的知識就介紹到這里了,還有一些更加深入的知識,比如:如何保證分布式環境中緩存數據的一致性、緩存數據的刷新、多級緩存時定制化緩存策略等等。這些都留待以后再學習和介紹吧!

參考資料:

Spring Cache: https://docs.spring.io/spring/docs/current/spring-framework-reference/integration.html#cacheCaffeine 緩存: https://www.jianshu.com/p/9a80c662dac4Layering-Cache:https://github.com/xiaolyuh/layering-cacheAlibaba JetCache: https://github.com/alibaba/jetcacheJetCache FAQ: https://github.com/alibaba/jetcache/wiki/FAQ_CN

以上就是詳解SpringBoot的三種緩存技術的詳細內容,更多關于SpringBoot 緩存技術的資料請關注好吧啦網其它相關文章!

標簽: Spring
相關文章:
成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
欧美jjzz| 美女视频黄 久久| 午夜一区二区三区在线观看| 午夜天堂精品久久久久| 精品国产免费一区二区三区四区| 韩国av一区二区三区在线观看| 日本丰满少妇一区二区三区| 一区二区三区四区在线播放| 亚洲精品1区| 自拍av一区二区三区| www..com久久爱| 欧美成人a视频| 成人精品视频一区| 精品成人私密视频| 99久久久免费精品国产一区二区| 欧美v亚洲v综合ⅴ国产v| 国产成人精品影视| 欧美一级生活片| 成人一区二区视频| 精品欧美一区二区久久| 国产成人午夜99999| 91精品国产色综合久久不卡蜜臀| 国产乱子伦一区二区三区国色天香| 欧美日韩免费观看一区三区| 国模娜娜一区二区三区| 欧美亚洲日本国产| 九九视频精品免费| 制服丝袜国产精品| 国产91对白在线观看九色| 日韩一区二区三区观看| 春色校园综合激情亚洲| 欧美v日韩v国产v| 91在线看国产| 亚洲国产成人自拍| 国产综合网站| 亚洲三级小视频| 国产一区二区三区黄| 视频在线观看一区| 欧美天堂一区二区三区| 国产传媒欧美日韩成人| 日韩视频永久免费| 91免费观看国产| 国产主播一区二区| 欧美电视剧在线观看完整版| 99精品欧美一区二区蜜桃免费| 欧美—级在线免费片| 夜夜嗨av一区二区三区中文字幕| 免费国产自线拍一欧美视频| 日本人妖一区二区| 欧美一级高清片| 91在线视频在线| 日韩伦理av电影| 亚洲一区二区三区精品在线观看| 日日夜夜一区二区| 欧美日韩亚洲丝袜制服| www.亚洲精品| 最新热久久免费视频| 男人的天堂亚洲在线| 精品一区二区在线看| 久久久三级国产网站| 亚洲精品一区二区三区av| 日韩黄色一级片| 日韩一区二区在线免费观看| 好看的亚洲午夜视频在线| 午夜不卡av免费| 9191久久久久久久久久久| 91麻豆精品一区二区三区| 亚洲乱码中文字幕| 欧美色倩网站大全免费| 色综合网色综合| 亚洲一区国产视频| 欧美精品视频www在线观看| 女人香蕉久久**毛片精品| 亚洲一区二区三区精品在线| 欧美日韩一区二区三区不卡| 91社区在线播放| 亚洲一区二区欧美日韩| 欧美一区永久视频免费观看| 国产精品啊啊啊| 男男成人高潮片免费网站| 亚洲精品一线二线三线| 国产精品毛片在线看| 国产一区二区三区免费在线观看| 欧美激情一区二区三区全黄| 91国偷自产一区二区使用方法| 成人国产精品免费网站| 一区二区三区中文在线观看| 欧美精品一二三| 好吊一区二区三区| 久久不见久久见免费视频1 | 自拍偷自拍亚洲精品播放| 欧美中文字幕一区二区三区亚洲| 91美女福利视频| 蜜臀av性久久久久蜜臀aⅴ四虎 | 国产欧美日韩综合一区在线播放| 国产在线国偷精品免费看| 欧美国产精品中文字幕| 欧美日韩一区中文字幕| 国语精品中文字幕| 国产一区二区美女诱惑| 亚洲精品亚洲人成人网在线播放| 欧美一区二区私人影院日本| 国产精品久久国产三级国电话系列| 国产露脸91国语对白| 一区二区久久久久久| 久久久高清一区二区三区| 欧美综合在线视频| 亚洲私人影院| 成人一区二区视频| 日本成人中文字幕在线视频| 中文字幕制服丝袜一区二区三区 | 欧美tk—视频vk| 一本大道久久a久久综合| 91麻豆免费视频| 久久国产三级精品| 亚洲视频资源在线| 精品久久一二三区| 欧洲精品视频在线观看| 精品1区2区3区4区| 粉嫩av一区二区三区| 天堂成人国产精品一区| 国产欧美日韩不卡免费| 在线成人av影院| 久久久久久9| 亚洲夫妻自拍| 欧美在线高清| 国产盗摄精品一区二区三区在线 | 国产欧美日韩伦理| 欧美/亚洲一区| 国产裸体歌舞团一区二区| 天天色综合成人网| 综合电影一区二区三区| 欧美va日韩va| 欧美日韩国产系列| 美女黄色成人网| 亚洲日产国产精品| 午夜精品久久| 成人免费观看视频| 激情av综合网| 美女视频免费一区| 亚州成人在线电影| 亚洲午夜精品17c| 亚洲精品网站在线观看| 亚洲国产激情av| 久久久久久夜精品精品免费| 欧美一区二区在线看| 欧美日韩极品在线观看一区| 欧美中日韩免费视频| 亚洲少妇一区| 在线欧美三区| 国产综合欧美在线看| 欧美福利一区二区三区| 成a人片亚洲日本久久| 国产精品一区在线观看你懂的| 青青草国产精品97视觉盛宴| 午夜电影网亚洲视频| 亚洲图片有声小说| 亚洲综合色网站| 一区二区三区在线视频播放| 亚洲人成伊人成综合网小说| 国产精品久久久久国产精品日日| 国产亚洲欧美日韩在线一区| 精品免费一区二区三区| 欧美成人一区二区三区| 日韩一区二区在线免费观看| 欧美一区二区不卡视频| 欧美一区二区黄| 日韩一级高清毛片| 欧美一区二区三区公司| 欧美一二三在线| 精品国产欧美一区二区| 欧美tk—视频vk| 精品动漫一区二区三区在线观看| 欧美一区二区三区视频免费 | 88在线观看91蜜桃国自产| 欧美欧美午夜aⅴ在线观看| 欧美视频一区二| 欧美日韩夫妻久久| 欧美日韩www| 欧美一区二区三区性视频| 91精品国产免费| 日韩欧美在线一区二区三区| 日韩免费视频线观看| 精品国产一区二区三区久久久蜜月| 91麻豆精品91久久久久久清纯 | 国产精品一区在线播放| 国产美女一区| 色女孩综合影院| 欧美性大战xxxxx久久久| 欧美日本韩国一区| 日韩一级片网站| 国产日韩欧美在线一区| 日韩一区在线看| 亚洲成人免费视| 日韩成人午夜精品| 国产一区二区三区精品视频| 成人av免费在线| 国产一区免费视频| 国产一区二区三区的电影 | 国产综合第一页|