Spring|因动态代理使用不注意导致的诡异现象

笔者在新的定时任务项目中,限定一个类只能写一个Job,类似于写脚本,一个Job一个脚本。对于简单的任务我们并不约定一定要有Service层,但在Job中我们可能需要将某些数据库操作放到事务中执行,为让注解事务生效,我们不能直接使用this调用事务方法。

调用本类事务方法有两种方式可以让注解事务生效:

  • 一是通过在类中注入自己,也就是循环依赖注入;

  • 二是在需要时再从bean工厂中获取bean

场景描述

假设现有类A,在类AmethodA方法中,先从Springbean工厂获取到类A的实例,再调用类AmethodB方法,这样做的目的是使事务生效。代码如下:

@Component
public class ProxyObjFieldNpe {

    @Value("${field_value}")
    private String fieldValue;

    public void methodA() {
        if (fieldValue == null) {
            System.out.println("methodA NPE...");
        }
        // 从bean工厂取,使AOP生效
        ProxyObjFieldNpe thisRef = OnionXxlJobApplicationContent.getBean(ProxyObjFieldNpe.class);
        // ......调用某些事务方法
        thisRef.methodB();
    }

    private void methodB() {
        if (fieldValue == null) {
            System.out.println("methodB NPE...");
        }
    }

}

外部调用methodA方法:proxyObjFieldNpe.methodA();

结果输出的是:"methodB NPE..."

为什么methodA方法获取到fieldValue字段的值不为空,而methodB方法获取到的fieldValue却为空呢?这就是笔者遇到的问题。细心的朋友,你有没有看出原因呢?

实际项目中调试的结果截图如下:

图中AutoCloseTimeoutOrderJob实例的字段都为空,这些字段都是声明自动注入的MapperService,不可能为空。但从调试结果我们可以看出,从bean工厂获取到的是AutoCloseTimeoutOrderJob的代理对象,并非AutoCloseTimeoutOrderJob

ProxyObjFieldNpe的例子中,我们从bean工厂获取到的也是ProxyObjFieldNpe的代理对象,该代理对象继承ProxyObjFieldNpe。因此,与上面截图一样,代理对象的字段都是NULL

外部调用ProxyObjFieldNpemethodA方法调用的是代理类的methodA方法,那为什么methodA方法拿到字段的值非空,而methodB方法拿到的是空值呢?

因为methodB方法被声明为private了,代理类没法重写该方法。因此thisRef.methodB();实际调用的是代理类父类的methodB方法。methodB方法中获取fieldValue获取的是代理类对象的,这就是methodA方法获取到fieldValue字段的值不为NULL,而methodB方法获取到fieldValue字段的值为NULL的原因。

“methodB方法中获取fieldValue获取的是代理类对象的 ",这句我们稍后从字节码层面理解。

问题:

  • 为什么methidB方法的访问标志是private,代理对象是ProxyObjFieldNpe的子类,却能调用其父类的methidB方法?

  • 为什么代理对象的字段为NULL?

为什么代理对象能调用父类的private方法?

因为调用访问标志为privatemethodB方法是在ProxyObjFieldNpe类的methodA方法中调用的,而不是在代理类的methodA方法中调用的,内部调用当然有访问权限。

代理类继承ProxyObjFieldNpe,外部调用代理类的methodA方法时,最终经过方法拦截器调用代理类父类的methodA方法,因此methodA方法中调用methodB方法实际上是在父类中调用的。

ProxyObjFieldNpemethodA方法编译后生成的字节码如下(部分):

   15: ldc           #6    // class com/wujiuye/test/ProxyObjFieldNpe
   17: invokestatic  #7    // Method com/wujiuye/test/OnionXxlJobApplicationContent.getBean:(Ljava/lang/Class;)Ljava/lang/Object;
   20: checkcast     #6    // class com/wujiuye/test/ProxyObjFieldNpe
   23: astore_1
   24: aload_1
   25: invokespecial #8    // Method com/wujiuye/test/ProxyObjFieldNpe.methodB:()V

偏移量为151720三条指令是:从bean工厂获取代理bean,并使用checkcast指令将代理对象类型强制转为父类类型。

偏移量为2425两条字节码实现调用methodB方法,非静态方法的第一个隐式参数为this引用,此处传的是代理类对象的引用,因此在methodB方法中,使用this(代理对象的引用)获取到的字段都是空的。

为什么代理对象的字段为NULL?

如果熟悉Spring Bean生命周期,那么就不难理解。

bean的创建过程如下:

  • 1、反射创建bean;

  • 2、为bean注入属性;

  • 3、调用*Aware接口的方法;

  • 4、调用BeanPostProcessorpostProcessBeforeInitialization方法;

  • 5、调用初始化方法,afterPropertiesSet或自定义的初始化方法;

  • 6、调用BeanPostProcessorpostProcessAfterInitialization方法;

代理对象是在上述步骤的第六步创建的,即调用某个BeanPostProcessorpostProcessAfterInitialization方法之后,返回代理对象,如果是单例对象,则会将该对象保存到bean工厂(容器)中。也就是说,bean工厂中存储的是代理对象。

下面两张图是我在项目中调试Spring代码的截图。(图中的小红点下方有个问号,这是条件断点,只有满足条件时才会停在断点处。条件的设置可右击小红点,在弹出框中输出条件,条件的编写与在代码中添加一个if语句是一样的。)

在调用BeanPostProcessorpostProcessAfterInitialization方法之前, bean还是原生的bean

在调用BeanPostProcessorpostProcessAfterInitialization方法之后,bean已经变成代理对象了。

因此,使用cglib生成的代理对象  (继承方式),在父类中,通过代理对象调用父类私有方法不会报错,但字段都是空的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

吴就业

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值