SpringBoot @Cacheable自定义KeyGenerator方式

网友投稿 814 2022-11-12

SpringBoot @Cacheable自定义KeyGenerator方式

SpringBoot @Cacheable自定义KeyGenerator方式

目录@Cacheable自定义KeyGenerator1. 概述2. MySimpleKey 类3. MyKeyGenerator 类4. 配置keyGeneratorSpring-Cache key设置第一种方式:手动设置第二种方式:自定义keyGenerator

@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 {

/**

* (非 javadoc)

*

* Title: generate

*

*

* @param target

* @param method

* @param params

* @return

* @see org.springframework.cache.interceptor.KeyGenerator#generate(java.lang.Object,

* java.lang.reflect.Method, java.lang.Object[])

*/

@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 |http://| 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

proxy-target-class="true" />

3、配置注解

如下图所示

proxy-target-class="true" />

3、配置注解

如下图所示

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:【无源物联网】物联网的下一个风口?
下一篇:Linux驱动实践:一起来梳理中断的前世今生(附代码)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~