前言
之前在写 –> 聊聊基于 jdk 实现的 spi 如何与 spring 整合实现依赖注入这篇文章的 demo 时,用到了动静代理,在进行调试,发现一个神奇的景象。如下图
代理对象变成 null,但不会有空指针异样
景象剖析
首先看下示例代理的外围实现逻辑
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {boolean canPass = preHandle(method,args);
Object result = null;
if(canPass){result = method.invoke(target,args);
afterCompletion(method,args);
}
return result;
}
这段逻辑当 canpass 为 false 时,result 会为 null。idea 开启调试, 调用对象时,默认会调用 toString 办法 ,当代理触发 invoke,因为 preHandle 找不到 toString 办法,会导致 canPass 为 false,从而触发 null 景象
口说无凭,咱们能够验证下,咱们对代理外围办法进行调整下
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {if("toString".equals(method.getName())){return "我是 toString 办法";}
boolean canPass = preHandle(method,args);
Object result = null;
if(canPass){result = method.invoke(target,args);
afterCompletion(method,args);
}
return result;
}
此时再进行 dubug,如下图
问题修复
1、办法一:禁用掉 idea 默认调用 toString 办法
2、办法二:在代理 invoke 办法中,增加如下如下代码片段
// 如果 Object 办法间接反射调用
if(Object.class.equals(method.getDeclaringClass())){return method.invoke(this, args);
}
这种解决思路,在 mybatis 实现的动静代理就有呈现
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
try {if (Object.class.equals(method.getDeclaringClass())) {return method.invoke(this, args);
} else if (method.isDefault()) {if (privateLookupInMethod == null) {return invokeDefaultMethodJava8(proxy, method, args);
} else {return invokeDefaultMethodJava9(proxy, method, args);
}
}
} catch (Throwable t) {throw ExceptionUtil.unwrapThrowable(t);
}
final MapperMethod mapperMethod = cachedMapperMethod(method);
return mapperMethod.execute(sqlSession, args);
}
上文摘自 mybatis 的 MapperProxy 代理片段
总结
多点好奇,多看源码