随笔 - 171  文章 - 0  评论 - 0  阅读 - 62466

代理对象执行私有方法导致注入的属性为null

代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
@Service
public class PerformanceServiceImpl implements IPerformanceService {
 
     private final static Logger logger = LoggerFactory.getLogger(PerformanceServiceImpl.class);
 
     @Resource
     PerformanceDao performanceDao;
 
     @Override
     @Transactional(rollbackFor = Exception.class)
     public void insertTest() throws Exception {
 
          ((PerformanceServiceImpl) AopContext.currentProxy()).insertD();
 
     }
 
     private void insertD() throws Exception {
          performanceDao.insertData();
     }
 
}

调用PerformanceServiceImpl 的insertTest方法时报错java.lang.NullPointerException。

debug发现在执行insertD方法时,performanceDao是null。

why?

  1. 在spring为PerformanceServiceImpl生成代理类时,由于insertD是私有方法,将不会被代理同理,私有方法也会导致事务失效。注:这里用的是cglib动态代理,图中明显看到Enchancer在生成代理类时,private方法会被过滤

     

     

  2. 通过java -cp .;%JAVA_HOME%/lib/sa-jdi.jar sun.jvm.hotspot.HSDB启动HotSpot VM调试器生成内存中的代理类,可以验证代理类并没有代理insertD方法

  3. 在insertTest调用insertD时,由于insertD方法没有被代理,所以是直接调用代理对象的insertD方法,而代理对象的performanceDao并没有被spring容器注入过

     

  4. 在普通的代理对象执行逻辑中,走完代理对象的逻辑后,最终还会调用切点的逻辑,也就是走被代理对象的被代理方法,被代理对象的performanceDao在spring创建bean的过程中被注入

     

     

posted on   zhengbiyu  阅读(72)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构
< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

点击右上角即可分享
微信分享提示