场景:
这是一个通知,目的是缓解数据库的压力,在查询数据时先查询redis,如果没有再去查询数据库,最后把结果返回给用户,同时异步的写入redis。
异步写入的过程:创建一个线程,创建匿名内部类实现runnable接口,这里在向内部类传参时,所传的参数需要final修饰
为什么要用final?
1.内部类传参时,需要用final修饰。在编译时,内部类会形成单独的一个class文件,这说明内部类和外部类是无关的,那就说明内部类需要外部类的参数时底层是会有一个复制的过程。
2.当执行一个方法时,该方法内的成员变量存在栈区,方法执行完成后,这些成员变量随之销毁。如果此时内部类调用外部类的成员变量时,就会出现问题。
3.final的作用,java中是使用共享对象传递来传值的,本质是把对象的地址复制一次,如果没有final,则这个变量可能会发生不可预料的修改。
@Around("saveRedis()")
public Object saveToRedis(ProceedingJoinPoint pjp) throws Throwable {
//返回结果
Object value = null;
//获取类名
String className = pjp.getTarget().getClass().getName();
//获取方法名
String methodName = pjp.getSignature().getName();
if (!isAddCache(className, methodName)) { //该路径禁止使用redis时
//跳过缓存返回结果
return pjp.proceed();
}
//获取参数
Object[] arguments = pjp.getArgs();
String key = getCacheKey(className, methodName, arguments);
try {
//判断是否有缓存
if (redisUtil.exists(key)) { //如果有,直接返回
return redisUtil.get(key);
}
//写入缓存
value = pjp.proceed();
if (value != null) {
final String tkey = key;
final Result tvalue = (Result) value;
new Thread(new Runnable() {
@Override
public void run() {
redisUtil.set(tkey, tvalue, defaultCacheExpireTime);
}
}).start();
}
} catch (Exception e) {
e.printStackTrace();
//如果起线程存redis的时候报错,则继续返回pjp.proceed()
if (value == null) {
return pjp.proceed();
}
}
return value;
}