聊聊在idea dubug模式下,动态代理类出现的null现象
前言
之前在写-->聊聊基于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代理片段
总结
多点好奇,多看源码
「其他文章」
- 聊聊如何基于spring @Cacheable扩展实现缓存自动过期时间以及即将到期自动刷新
- 聊聊基于docker部署的mysql如何进行数据恢复
- 聊聊如何验证线上的版本是符合预期的版本
- 聊聊如何让你的业务代码具有可扩展性
- spring事务失效的几种场景以及原因
- 聊聊自定义SPI如何与sentinel整合实现熔断限流
- 聊聊如何实现一个支持键值对的SPI
- 聊聊springboot项目如何实现自定义actuator端点
- 聊聊使用lombok @Builder踩到的坑
- 笑话
- 聊聊如何自定义实现maven插件
- 聊聊在idea dubug模式下,动态代理类出现的null现象
- 聊聊基于jdk实现的spi如何与spring整合实现依赖注入
- 笑话
- 聊聊springcloud项目同时存在多个注册中心客户端采坑记
- feign请求返回值反序列LocalDateTime异常记录
- 笑话
- 笑话
- 笑话
- 笑话