首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >聊聊在idea dubug模式下,动态代理类出现的null现象

聊聊在idea dubug模式下,动态代理类出现的null现象

作者头像
lyb-geek
发布2021-09-23 15:53:08
发布2021-09-23 15:53:08
40100
举报
文章被收录于专栏:Linyb极客之路Linyb极客之路
运行总次数:0
01

前言

之前在写-->聊聊基于jdk实现的spi如何与spring整合实现依赖注入这篇文章的demo时,用到了动态代理,在进行调试,发现一个神奇的现象。如下图

代理对象变成null,但不会有空指针异常

02

现象分析

首先看下示例代理的核心实现逻辑

代码语言:javascript
代码运行次数:0
运行
复制
@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现象

口说无凭,我们可以验证下,我们对代理核心方法进行调整下

代码语言:javascript
代码运行次数:0
运行
复制
@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,如下图

03

问题修复

01

方法一:禁用掉idea默认调用toString方法

02

方法二:在代理invoke方法中,添加如下如下代码片段

代码语言:javascript
代码运行次数:0
运行
复制
if(Object.class.equals(method.getDeclaringClass())){
            return method.invoke(this, args);
        }

这种解决思路,在mybatis实现的动态代理就有出现

代码语言:javascript
代码运行次数:0
运行
复制
@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代理片段

04

总结

多点好奇,多看源码

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2021-08-29,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Linyb极客之路 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档