@Cacheable自定义KeyGenerator
1. 概述
SpringBoot 使用 @Cacheable 可以方便的管理缓存数据,在不指定 key 属性的情况下,默认使用 SimpleKeyGenerator 生成 key。除此之外,我们也可以自定义实现 KeyGenerator 接口,生成自己的 key 名称策略。
2. MySimpleKey 类
MySimpleKey类的作用是存放参数数据,必须实现equals、hashCode。如果需要自定义key格式,同样需要实现toString接口,下面的例子是把参数用逗号分隔。
public class MySimpleKey implements Serializable {
public static final MySimpleKey EMPTY = new MySimpleKey(new Object[0]);
private final Object[] params;
private transient int hashCode;
public MySimpleKey(Object... elements) {
Assert.notNull(elements, "Elements must not be null");
this.params = (Object[])elements.clone();
this.hashCode = Arrays.deepHashCode(this.params);
}
public boolean equals(@Nullable Object other) {
return this == other || other instanceof MySimpleKey && Arrays.deepEquals(this.params, ((MySimpleKey)other).params);
}
public final int hashCode() {
return this.hashCode;
}
public String toString() {
return StringUtils.arrayToCommaDelimitedString(this.params);
}
}
3. MyKeyGenerator 类
MyKeyGenerator 实现 KeyGenerator 的接口,里面只有一个 generate 方法
public class MyKeyGenerator implements KeyGenerator {
@Override
public Object generate(Object o, Method method, Object... objects) {
if (objects.length == 0) {
return MySimpleKey.EMPTY;
} else {
if (objects.length == 1) {
Object param = objects[0];
if (param != null && !param.getClass().isArray()) {
return param;
}
}
return new MySimpleKey(objects);
}
}
}
定义MyKeyGenerator Bean:
@Component
public class MyRedisConf {
@Bean
public MyKeyGenerator myKeyGenerator(){
return new MyKeyGenerator();
}
}
4. 配置keyGenerator
在 @Cacheable 配置 keyGenerator 属性,值就是前面配置的Bean名称
@Override
@Cacheable(value = {"REDIS:GETSTRING3"}, keyGenerator = "myKeyGenerator")
public String getString3(String tag, String name) {
return tag + " " + name;
}
测试结果如下,tag、name 参数确实以逗号分隔
127.0.0.1:6379[5]> KEYS *
1) "REDIS:GETSTRING3::hello,zhangsan"
Spring-Cache key设置
第一种方式:手动设置
为了便于key的不重复,我们可以手动设置key有类名、方法名、参数等组合
属性名称 |
描述 |
示例 |
methodName |
当前方法名 |
#root.methodName |
method |
当前方法 |
#root.method.name |
target |
当前被调用的对象 |
#root.target |
targetClass |
当前被调用的对象的class |
#root.targetClass |
args |
当前方法参数组成的数组 |
#root.args[0] |
caches |
当前被调用的方法使用的Cache |
#root.caches[0].name |
key = "#root.targetClass.simpleName+':'+#root.methodName+':'+#param"
如下图所示
第二种方式:自定义keyGenerator
1、自定义CacheKeyGenerator 实现KeyGenerator
public class CacheKeyGenerator implements KeyGenerator {
@Override
public Object generate(Object target, Method method, Object... params) {
StringBuilder key = new StringBuilder();
key.append(target.getClass().getSimpleName()).append(":").append(method.getName()).append(":");
if (params.length == 0) {
return key.toString();
}
for (int i = 0; i < params.length; i++) {
Object param = params[i];
if (param == null || param instanceof LogableParam) {
del(key);
} else if (ClassUtils.isPrimitiveArray(param.getClass())) {
int length = Array.getLength(param);
for (int j = 0; j < length; j++) {
key.append(Array.get(param, j));
key.append(',');
}
} else if (ClassUtils.isPrimitiveOrWrapper(param.getClass()) || param instanceof String) {
key.append(param);
} else {
key.append(param.toString());
}
key.append('-');
}
del(key);
return key.toString();
}
private StringBuilder del(StringBuilder stringBuilder) {
if (stringBuilder.toString().endsWith("-")) {
stringBuilder.deleteCharAt(stringBuilder.length() - 1);
}
return stringBuilder;
}
}
2、配置xml
<cache:annotation-driven cache-manager="cacheManager" key-generator="cacheKeyGenerator"
proxy-target-class="true" />
<bean id="cacheKeyGenerator" class="com.tensoon.util.CacheKeyGenerator"></bean>
3、配置注解
如下图所示
以上为个人经验,希望能给大家一个参考,也希望大家多多支持编程网。