关于IDEA:聊聊在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代理片段

总结

多点好奇,多看源码

评论

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

这个站点使用 Akismet 来减少垃圾评论。了解你的评论数据如何被处理