利用反射注册SpringCache的RedisCacheManager缓存信息
项目开发中,SpringCache是一个非常方便的工具,但是在配置信息注册时,用枚举方式可以满足遍历,但却无法应用在@Cacheable注解里,因此可以通过静态类的方式,借助反射完成缓存信息注册。
配置类如下:
package com.zora.cloud.demo.config;
import com.alibaba.fastjson.support.spring.GenericFastJsonRedisSerializer;
import lombok.extern.slf4j.Slf4j;
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 org.springframework.data.redis.serializer.StringRedisSerializer;
import java.lang.reflect.Field;
import java.time.Duration;
import java.util.HashMap;
import java.util.Map;
/**
* <h3>metaapp-gamex</h3>
* <h4>com.metaapp.cloud.gamex.config</h4>
* <p>SpringCache配置</p>
*
* @author Yuhan.Ji
* @since 2020.03.23
*/
@Slf4j
@Configuration
public class CacheConfig {
private String appName = "Demo";
/**
* 当没有配置缓存失效时间情况下的默认ttl
*/
public static final int DEFAULT_EXPIRE_SECOND = 180;
// 用于SpringBoot 2.0和2.1的配置方式
@Bean
public RedisCacheManager builder(RedisConnectionFactory factory) {
try {
// 这里的参数填充对应的静态属性类
Map<String, Integer> configMap = ReflectUtil.getCacheTtlConfigMap(Constants.class);
RedisCacheManager.RedisCacheManagerBuilder builder = RedisCacheManager.RedisCacheManagerBuilder
.fromConnectionFactory(factory);
if (!configMap.isEmpty()) {
Map<String, RedisCacheConfiguration> configurationMap = new HashMap<>();
for (String cacheName : configMap.keySet()) {
configurationMap.
put(cacheName, RedisCacheConfiguration
.defaultCacheConfig()
.computePrefixWith(name -> appName + ":" + name + ":")
.serializeKeysWith(RedisSerializationContext.SerializationPair.fromSerializer(new StringRedisSerializer()))
.serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(new GenericFastJsonRedisSerializer()))
.entryTtl(Duration.ofSeconds(configMap.get(cacheName))));
}
builder.withInitialCacheConfigurations(configurationMap);
log.info("MetaApp Spring Cache Redis : 缓存配置为:{}", configMap.toString());
}
return builder.build();
} catch (IllegalAccessException illegalEx) {
log.error("SpringCache RedisCacheManagerBuilderCustomizer配置失败,启动终止。请检查RedisCacheManagerBuilder类中ReflectUtil.getCacheTtlConfigMap()部分", illegalEx);
System.exit(500);
return null;
}
}
// // 用于SpringBoot 2.2 的配置方式
// @Bean
// public RedisCacheManagerBuilderCustomizer redisCacheManagerBuilderCustomizer() {
// return builder -> {
// try {
// // 这里的参数填充对应的静态属性类
// Map<String, Integer> configMap = ReflectUtil.getCacheTtlConfigMap(Constants.class);
// if (!configMap.isEmpty()) {
// Map<String, RedisCacheConfiguration> configurationMap = new HashMap<>();
// for (String cacheName : configMap.keySet()) {
// configurationMap.
// put(cacheName, RedisCacheConfiguration
// .defaultCacheConfig()
// .computePrefixWith(name -> appName + ":" + name + ":")
// .serializeKeysWith(RedisSerializationContext.SerializationPair.fromSerializer(new StringRedisSerializer()))
// .serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(new GenericFastJsonRedisSerializer()))
// .entryTtl(Duration.ofSeconds(configMap.get(cacheName))));
// }
// builder.withInitialCacheConfigurations(configurationMap);
// log.info("Redis缓存配置为:{}",configMap.toString());
// }
// } catch (IllegalAccessException illegalEx) {
// log.error("SpringCache RedisCacheManagerBuilderCustomizer配置失败,启动终止。请检查RedisCacheManagerBuilder类中ReflectUtil.getCacheTtlConfigMap()部分", illegalEx);
// System.exit(500);
// }
// };
// }
static class ReflectUtil {
public static final String CACHE_NAME_PREFIX = "CACHE_NAME_";
public static final String CACHE_TTL_PREFIX = "CACHE_TTL_";
static <T> Map<String, Integer> getCacheTtlConfigMap(Class<T> clazz) throws IllegalAccessException {
Map<String, Integer> cacheMapWithKeyTtl = new HashMap<>();
Field[] fields = clazz.getFields();
Map<String, String> cacheNames = new HashMap<>(fields.length);
// 配置cacheName
for (Field field : fields) {
if (field.getName().contains(CACHE_NAME_PREFIX)) {
if (field.getType().equals(String.class)) {
String lookingForTtlName = field.getName().replace(CACHE_NAME_PREFIX, CACHE_TTL_PREFIX);
String value = (String) field.get(clazz);
cacheNames.put(lookingForTtlName, value);
} else {
throw new IllegalAccessException("In constants configuration, the type of prefix of cache must be String. Illegal in " + clazz.getName() + "." + field.getName() + ".");
}
}
}
// 配置过期时间
for (Field field : fields) {
if (field.getName().contains(CACHE_TTL_PREFIX)) {
if (field.getType().equals(int.class) || field.getType().equals(Integer.class)) {
if (cacheNames.containsKey(field.getName())) {
cacheMapWithKeyTtl.put(cacheNames.get(field.getName()), field.getInt(clazz));
}
} else {
throw new IllegalAccessException("In constants configuration, the type of prefix of ttl must be int or Integer. Illegal in " + clazz.getName() + "." + field.getName() + ".");
}
}
}
// 配置默认过期时间
for (String cacheNameKey : cacheNames.keySet()) {
String cacheName = cacheNames.get(cacheNameKey);
if (!cacheMapWithKeyTtl.containsKey(cacheNames.get(cacheNameKey))) {
log.warn("===========================Spring Cache Config 配置信息错误=========================");
log.warn("Spring Cache Config 配置信息错误:{}文件中的缓存{}没有查询到对应失效时间的配置信息,将按默认值180秒执行。", clazz.getName(), cacheName);
log.warn("=============================== Zora Cache Config ================================");
cacheMapWithKeyTtl.put(cacheName, DEFAULT_EXPIRE_SECOND);
}
}
return cacheMapWithKeyTtl;
}
}
}
利用反射注册SpringCache的RedisCacheManager缓存信息的更多相关文章
- java封装返回结果应用 利用反射机制 返回结果以及错误信息
//此方法是自己琢磨的,可能有漏洞--符合我的系统目前的需求,大家可以借鉴public ResponseMessage exec (String funname, Object[] params,Ob ...
- 利用反射来实现获取成员的指定特性(Attribute)信息
在开发过程中,我们经常需要自定义一些特性,来辅助我们完成对对象或者枚举进行管理.我们需要知道如何获取对象使用的特性信息. 以下举个学习用的例子. 我们自定义一个特性类,这个特性设置在一个数据段内是否执 ...
- c#反射机制学习和利用反射获取类型信息
反射(Reflection)是.NET中的重要机制,通过放射,可以在运行时获得.NET中每一个类型(包括类.结构.委托.接口和枚举等)的成员,包括方法.属性.事件,以及构造函数等.还可以获得每个成员的 ...
- C# winform利用反射和自定义特性加载功能模块(插件式开发)
由于在实际的工作中, 碰见这样的一个问题: 一个软件, 销售给A客户 他需要所有功能, 但是销售给B客户, 他只需要其中的一部分, 1.如果我们在实际的开发过程中, 没有把一些功能模块区分开来的话, ...
- 利用反射快速给Model实体赋值 使用 Task 简化异步编程 Guid ToString 格式知多少?(GUID 格式) Parallel Programming-实现并行操作的流水线(生产者、消费者) c# 无损高质量压缩图片代码 8种主要排序算法的C#实现 (一) 8种主要排序算法的C#实现 (二)
试想这样一个业务需求:有一张合同表,由于合同涉及内容比较多所以此表比较庞大,大概有120多个字段.现在合同每一次变更时都需要对合同原始信息进行归档一次,版本号依次递增.那么我们就要新建一张合同历史表, ...
- 【转】利用反射快速给Model实体赋值
原文地址:http://blog.csdn.net/gxiangzi/article/details/8629064 试想这样一个业务需求:有一张合同表,由于合同涉及内容比较多所以此表比较庞大,大概有 ...
- 静态代理和利用反射形成的动态代理(JDK动态代理)
代理模式 代理模式的定义:为其他对象提供一种代理以控制对这个对象的访问.在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用. 静态代理 1.新建 ...
- 利用反射实现通用的excel导入导出
如果一个项目中存在多种信息的导入导出,为了简化代码,就需要用反射实现通用的excel导入导出 实例代码如下: 1.创建一个 Book类,并编写set和get方法 package com.bean; p ...
- [.net 面向对象程序设计进阶] (21) 反射(Reflection)(下)设计模式中利用反射解耦
[.net 面向对象程序设计进阶] (21) 反射(Reflection)(下)设计模式中利用反射解耦 本节导读:上篇文章简单介绍了.NET面向对象中一个重要的技术反射的基本应用,它可以让我们动态的调 ...
随机推荐
- .NetCore Docker一次记录
1:项目添加docker支持 2:定位到项目主目录 按住shift,鼠标右键,打开powershell,输入命令 dotnet publish 此时会在目录 bin\Debug\netcoreapp2 ...
- 使用 .NET 5 体验大数据和机器学习
翻译:精致码农-王亮 原文:http://dwz.win/XnM .NET 5 旨在提供统一的运行时和框架,使其在各平台都有统一的运行时行为和开发体验.微软发布了与 .NET 协作的大数据(.NET ...
- Shell脚本常用命令整理
该笔记主要整理了一些常见的脚本操作命令,大致如下(持续补充中): 1. while.for循环 1. while.for循环 #!/bin/bash # while循环 v_start_date=${ ...
- waitpid()系统调用学习
waitpid()的头文件 #include <sys/types.h> #include <sys/wait.h> pid_t waitpid(pid_t pid,int ...
- 浅析TCP协议---转载
https://cloud.tencent.com/developer/article/1150971 前言 说到TCP协议,相信大家都比较熟悉了,对于TCP协议总能说个一二三来,但是TCP协议又是一 ...
- 聊一聊sockmap 以及ebpf 实例演示
eBPF实质上是一个内核注入技术 用户态可以用C来写运行的代码,再通过一个Clang&LLVM的编译器将C代码编译成BPF目标码: 用户态通过系统调用bpf()将BPF目标码注入到内核当中,并 ...
- DB2 建表,添加字段,删除字段,修改字段等常用操作
转载:http://blog.sina.com.cn/s/blog_67aaf4440100v01p.html,稍作修改. --创建数据库 create database Etp; --连接数据库 c ...
- ubuntu 18.04安装RTL8821CE无线网卡驱动
疫情期间闲下来无聊,把办公室的旧机器装了ubuntu,但是无法连接无线网. 打开终端 #查看无线网卡信息. -i 是不区分大小写 tjj@ubuntu:~/Documents$ lspci | gre ...
- python any,call,init,下划线知识汇总
python补充 any() [来自菜鸟教程] any() 函数用于判断给定的可迭代参数 iterable 是否全部为 False,则返回 False,如果有一个为 True,则返回 True. 元素 ...
- 应用程序-特定 权限设置并未向在应用程序容器不可用 SID (不可用)中运行的地址 LocalHost (使用 LRPC) 中的用户...的 COM 服务器应用程序的 本地 激活 权限。此安全权限可以使用组件服务管理工具进行修改。
很久以前发现我们的业务服务器上出现一个System的系统严重错误,查找很久都没有找到解决办法,今日再次查看服务器发现报错更频繁,于是就搜集各种资料进行查找解决办法,终于找到了一个解决办法. 错误截图介 ...