• 聊聊在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方法

    image.png

    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代理片段

    总结

    多点好奇,多看源码

  • 相关阅读:
    只要三步,使用html5+js实现像素风头像生成器
    按Ctrl+Enter发送的实现
    “放到桌面”的Servlet实现
    从tom大叔那想着拿书的,呵呵。
    也写dateUtil.js
    智习室
    零基础爬虫课,不会编程也能做爬虫
    1小时教你学会如何采集微博数据:0基础小白也能轻松学会!
    TransactionScope 分布式事务配置
    centos7创建共享文件夹
  • 原文地址:https://www.cnblogs.com/linyb-geek/p/15178937.html
Copyright © 2020-2023  润新知