【Bug记录】调用Controller中的方法的Service为null

问题

今天写代码的时候遇到了一个奇怪的Bug,在Controller中调用某个方法时突然报空指针异常,打断点时发现service为null,并尝试了数种注入方式,发现均为null。

原因分析

首先去检查了一下Service上的注解,发现是有 @Service 注解的,因为使用了Lombok的 @RequiredArgsConstructor 注解1,所以在想是不是Lombok的原因,后面去掉该注解,分别尝试了使用 @Autowired 注入、@Resource 注入以及构造器注入,调用时发现仍为null。

上网查询资料后发现关注点错了,既然Service添加了对应注解,且也尝试了不同的注入方法,如果其他方法也均是使用这种方式注入且没出问题的,那就说明问题出在方法上。

在检查方法时发现,在Controller层,方法定义时使用了private 修饰,而不是正常的 public,如下:

查阅资料后发现:容器在扫描Bean并生成代理类的时候,只会生成被 publicprotected 修饰的方法的代理类,被 private 修饰的方法并不会生成代理类。

Q: 那为什么我们的类是使用 public 方法修饰的,此处使用 private 修饰方法就无法获取到Bean了呢?
A: 因为属性的注入也是在代理类中完成的,被 publicprotected 修饰的方法获取到的属性也是已经完成注入的属性,而 private 修饰的方法获取的是还未完成注入的属性,因此获取Bean时为null。

1. @RequiredArgsConstructor注解:加了该注解的类会在代码编译时,自动给该类加上一个构造器注入,无需使用 @Autowired 或 @Resource 注解,其生效条件为必须由 final 所修饰。