原文链接:【Bug 记录】调用 Controller 中的方法的 Service 为 null
问题
今天写代码的时候遇到了一个奇怪的Bug,在Controller中调用某个方法时突然报空指针异常,打断点时发现service为null,并尝试了数种注入方式,发现均为null。
原因分析
首先去检查了一下Service上的注解,发现是有 @Service
注解的,因为使用了Lombok的 @RequiredArgsConstructor
注解1,所以在想是不是Lombok的原因,后面去掉该注解,分别尝试了使用 @Autowired 注入、@Resource 注入以及构造器注入,调用时发现仍为null。
上网查询资料后发现关注点错了,既然Service添加了对应注解,且也尝试了不同的注入方法,如果其他方法也均是使用这种方式注入且没出问题的,那就说明问题出在方法上。
在检查方法时发现,在Controller层,方法定义时使用了private
修饰,而不是正常的 public
,如下:
查阅资料后发现:容器在扫描Bean并生成代理类的时候,只会生成被 public
和 protected
修饰的方法的代理类,被 private
修饰的方法并不会生成代理类。
Q: 那为什么我们的类是使用 public
方法修饰的,此处使用 private
修饰方法就无法获取到Bean了呢?
A: 因为属性的注入也是在代理类中完成的,被 public
和 protected
修饰的方法获取到的属性也是已经完成注入的属性,而 private
修饰的方法获取的是还未完成注入的属性,因此获取Bean时为null。
@RequiredArgsConstructor注解:加了该注解的类会在代码编译时,自动给该类加上一个构造器注入,无需使用 @Autowired 或 @Resource 注解,其生效条件为必须由 final 所修饰。 ↩︎