beanpostprocessor使用场景_Spring因动态代理使用不注意导致的诡异现象
笔者在新的定时任务项目中,限定一个类只能写一个Job
,类似于写脚本,一个Job
一个脚本。对于简单的任务我们并不约定一定要有Service
层,但在Job
中我们可能需要将某些数据库操作放到事务中执行,为让注解事务生效,我们不能直接使用this
调用事务方法。
调用本类事务方法有两种方式可以让注解事务生效:
一是通过在类中注入自己,也就是循环依赖注入;
二是在需要时再从
bean
工厂中获取bean
;
场景描述
假设现有类A
,在类A
的methodA
方法中,先从Spring
的bean
工厂获取到类A
的实例,再调用类A
的methodB
方法,这样做的目的是使事务生效。代码如下:
@Componentpublic 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
实例的字段都为空,这些字段都是声明自动注入的Mapper
与Service
,不可能为空。但从调试结果我们可以看出,从bean
工厂获取到的是AutoCloseTimeoutOrderJob
的代理对象,并非AutoCloseTimeoutOrderJob
。
在ProxyObjFieldNpe
的例子中,我们从bean
工厂获取到的也是ProxyObjFieldNpe
的代理对象,该代理对象继承ProxyObjFieldNpe
。因此,与上面截图一样,代理对象的字段都是NULL
。
外部调用ProxyObjFieldNpe
的methodA
方法调用的是代理类的methodA
方法,那为什么methodA
方法拿到字段的值非空,而methodB
方法拿到的是空值呢?
因为methodB
方法被声明为private
了,代理类没法重写该方法。因此thisRef.methodB();
实际调用的是代理类父类的methodB
方法。methodB
方法中获取fieldValue
获取的是代理类对象的,这就是methodA
方法获取到fieldValue
字段的值不为NULL
,而methodB
方法获取到fieldValue
字段的值为NULL
的原因。
“methodB
方法中获取fieldValue
获取的是代理类对象的 ",这句我们稍后从字节码层面理解。
问题:
为什么
methidB
方法的访问标志是private
,代理对象是ProxyObjFieldNpe
的子类,却能调用其父类的methidB
方法?为什么代理对象的字段为
NULL
?
为什么代理对象能调用父类的private方法?
因为调用访问标志为private
的methodB
方法是在ProxyObjFieldNpe
类的methodA
方法中调用的,而不是在代理类的methodA
方法中调用的,内部调用当然有访问权限。
代理类继承ProxyObjFieldNpe
,外部调用代理类的methodA
方法时,最终经过方法拦截器调用代理类父类的methodA
方法,因此methodA
方法中调用methodB
方法实际上是在父类中调用的。
ProxyObjFieldNpe
的methodA
方法编译后生成的字节码如下(部分):
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
偏移量为15
、17
、20
三条指令是:从bean
工厂获取代理bean
,并使用checkcast
指令将代理对象类型强制转为父类类型。
偏移量为24
、25
两条字节码实现调用methodB
方法,非静态方法的第一个隐式参数为this
引用,此处传的是代理类对象的引用,因此在methodB
方法中,使用this
(代理对象的引用)获取到的字段都是空的。
为什么代理对象的字段为NULL?
如果熟悉Spring Bean
生命周期,那么就不难理解。
bean
的创建过程如下:
1、反射创建
bean
;2、为
bean
注入属性;3、调用
*Aware
接口的方法;4、调用
BeanPostProcessor
的postProcessBeforeInitialization
方法;5、调用初始化方法,
afterPropertiesSet
或自定义的初始化方法;6、调用
BeanPostProcessor
的postProcessAfterInitialization
方法;
代理对象是在上述步骤的第六步创建的,即调用某个BeanPostProcessor
的postProcessAfterInitialization
方法之后,返回代理对象,如果是单例对象,则会将该对象保存到bean
工厂(容器)中。也就是说,bean
工厂中存储的是代理对象。
下面两张图是我在项目中调试Spring
代码的截图。(图中的小红点下方有个问号,这是条件断点,只有满足条件时才会停在断点处。条件的设置可右击小红点,在弹出框中输出条件,条件的编写与在代码中添加一个if
语句是一样的。)
在调用BeanPostProcessor
的postProcessAfterInitialization
方法之前, bean
还是原生的bean
。
在调用BeanPostProcessor
的postProcessAfterInitialization
方法之后,bean
已经变成代理对象了。
因此,使用cglib
生成的代理对象 (继承方式),在父类中,通过代理对象调用父类私有方法不会报错,但字段都是空的。
公众号:Java艺术扫码关注最新动态
beanpostprocessor使用场景_Spring因动态代理使用不注意导致的诡异现象相关推荐
- Spring AOP的实现原理及应用场景(通过动态代理)
点击关注公众号,利用碎片时间学习 AOP的作用 作用:在不修改源代码的情况下,可以实现功能的增强. 传统的纵向体系代码复用: 横向抽取机制(AOP思想): AOP 思想:基于代理思想,对原来目标对象, ...
- Java进阶 | Proxy动态代理机制详解
一.Jvm加载对象 在说Java动态代理之前,还是要说一下Jvm加载对象的过程,这个依旧是理解动态代理的基础性原理: Java类即源代码程序.java类型文件,经过编译器编译之后就被转换成字节代码.c ...
- 设计模式之代理:手动实现动态代理,揭秘原理实现
作者:青石路 cnblogs.com/youzhibing/p/10464274.html 什么是代理模式 所谓代理,就是一个人或者一个机构代表另一个人或者另一个机构采取行动.在一些情况下,一个客户不 ...
- java动态代理详解及实际应用
重点教你如何应用 先回顾一下代理模式 使用场景1:动态代理+反射 实现hook技术 AOP(面向切面编程) 源码下载 先回顾一下代理模式 代理模式分动态代理和静态代理,写起来也比较简单,先上代码: p ...
- 接住喽????,送你个装逼的技能: JDK动态代理
今天讲一个比较深层的知识点:JDK动态代理,这是个可以让小白在大咖面前装逼的神器,顺便送你一个代理模式的温习机会. 代理模式场景 为了引出动态代理的用法,我们先看看代理设计模式,这能让你了解JDK动态 ...
- 面试篇之HR问什么是静态代理?什么是动态代理?
何为代理? Java中的代理,开源理解为通过代理去访问实际的目标对象,比如呢?我们平常买卖二手车的中间商,就可以看作一个代理类,不过你也可以直接去和二手车的主人买卖. 那这种情况,在Java中就被称之 ...
- 动态代理是什么?应用场景?
动态代理是什么?应用场景? 动态代理:在运行时,创建目标类,可以调用和扩展目标类的方法. Java 中实现动态的方式:JDK 中的动态代理 和 Java类库 CGLib. 应用场景如: 统计每个 ap ...
- 什么是动态代理?应用场景?
动态代理是什么?应用场景? 动态代理:在运行时,创建目标类,可以调用和扩展目标类的方法. Java 中实现动态的方式:JDK 中的动态代理 和 Java类库 CGLib. 应用场景如: 统计每个 ap ...
- spring的aop的动态代理机制都有哪些_Spring学习(4):Spring AOP
Spring AOP说明 AOP(Aspect Oriented Pragraming)面向切面编程,AOP采用横向抽取机制,取代了传统纵向继承体系的重复性代码(性能监视.事务管理.安全检查.缓存). ...
最新文章
- 哪些品牌开始连接鸿蒙系统,华为鸿蒙系统硬件生态品牌升级为 HarmonyOS Connect
- struts2处理请求流程详解
- Class Imbalance Problem
- yum 如何知道从哪里下载包?
- iframe高度自适应(IE6+、FF、Opera、Chrome等测试通过)
- adc0809引脚图及功能_80C51单片机的引脚及其功能介绍
- 图片少量显示 9张一下 类似微信,微博客户端
- [趣谈网络协议学习] 17 P2P协议:我下小电影, 99%急死你
- 防止电脑自动锁屏(Windows系统)
- php error unexpected,PHP异常Parse error: syntax error, unexpected错误解决方法
- Office Excel2010保存新文件时出现未响应或者卡死的解决办法
- 云存储Storj简要原理
- 容器编排-Docker Compose
- Spring Boot 探索之旅(二)—— Hello World
- python安装第三方库遇到 ERROR: Command errored out with exit status 1:
- CINTA作业七:同态
- hy3208中文参数_HY-MOSFET-产品目录表
- 微信个性签名服务器维护,今天更新了微信个性签名(重启之路79)
- 好看的数据可视化图片都是用什么做的? | 数答
- idea的热键占用问题解决方法
热门文章
- java swing 外观框架_Swing外观框架BeautyEye使用
- 【Matlab】求解积分方程的数值解
- 【控制】《多无人机协同控制技术》周伟老师-第8章-危险状态下的无人机编队运动控制策略
- 【Matlab 图像】bwlabel() 连通域及图像分割
- 1.4 满足和优化指标-深度学习第三课《结构化机器学习项目》-Stanford吴恩达教授
- 在Windows环境下编译cocos2d-x-3.0
- Ubuntu 12.04 下编译Android 4.0.3
- 硬核创客DIY遥控氦气球飞行器——会飞的鲨鱼的诞生的全过程
- 利用RC网络降低可调节LDO输出噪声
- source insight 4.0的基本使用方法