自己new出来的bean中被@Autowired注解修饰的属性报空指针异常
异常描述
原本我再测试RabbitMQ的发送程序,里面用到了一个AmqpTemplate
接口,用了@Autowired
注解。但是当我使用AmqpTemplate
的convertAndSend()
方法时却总报空指针异常 :java.lang.NullPointException
错误代码
修改过的可重现问题代码:
@Component
public class Test
{ //修改过的重现问题代码
@Autowired
AmqpTemplate amqpTemplate;
public Test() {
System.out.println("constructer: amqpTemplate"+Objects.isNull(amqpTemplate)+"#############");
}
@PostConstruct //该注解的意思是会在Autowired注入之后执行
public void init(){
System.out.println("init:amqpTemplate"+Objects.isNull(amqpTemplate)+"#############");
}
public void send(){ //外部调用的方法
System.out.println("send: amqpTemplate"+Objects.isNull(amqpTemplate)+"#############");
}
}
public class TestController{
@GetMapper("...省略")
public void testMethod(){
Test test = new Test(); //通过自行new,而不是在Controller中@Autowired来修饰
test.send();
}
}
结果
//报空指针异常
constructer: amqpTemplate true+"#############"
init: amqpTemplate false+"#############"
constructer: amqpTemplate true+"#############"
send: amqpTemplate true+"#############"
异常分析
需要了解的前提:
- 构造方法和@Autowired和@PostConstruct的执行顺序是:constructor > @Autowired > @PostConstruct
- @PostConstruct修饰的方法会在@Autowired注入后执行
情况是这样的:
- 我们在Test类中声明了@Component,所以项目启动时,Spring会构建一次Test类,放入Spring Bean容器
- 我们在Test类中使用@Autowired注解的方式来注入属性,即通过Spring的方式来获得AmqpTemplate实例
- 我们在TestController类中没有使用@Autowired的方式来获得Test实例,而是自行new
异常分析:
我们看到结果输出了4句话
constructer: amqpTemplate true+"#############"
init: amqpTemplate false+"#############"
constructer: amqpTemplate true+"#############"
send: amqpTemplate true+"#############"
- 第一句话肯定是
@Component
注解的作用,即spring构建了Test类的实例,放入Spring Bean容器中,此时的amqpTemplate
还未被注入,所以是null - 第二句是Spring容器构建的时候,触发了
Init
方法,此时的amqpTemplate
已经经过了@Autowried
注入,所以是false,已经不是null了。Spring容器中的amqpTemplate
被注入到Test实例中的amqpTemplate
属性中。 - 第三句开始就是重点了,是我们在
TestController
执行的new Test()
方法,人为触发了Test的构造函数,不是Spring容器。此时的amqpTemplate
实例,我们发现是true。不过在构造函数阶段是null也正常。 - 第四句是紧跟构造函数和
@Autowired
注入的init
方法触发的。我们发现amqpTemplate
是否为空还是一个true,也就是说@Autowired
没有注入成功。
好了,我们发现问题所在了,也就是在自己new一个Test实例的时候,Test类的amqpTemplate属性并没有被注入
结论
为什么会空指针异常呢?这的确是一个好问题,其实也很傻,因为本质的问题是在于对Spring的机制不够了解。
原因:
原因就是一个内含有@Autowired注解的Bean,我们必须要通过Spring的方式来获得这个Bean,而不能自己new一个。因为自己new出来的bean无法通过Spring容器来获得自己需要的bean.可以简单的理解为,要就全都使用Spring容器来管理,要不都不用,你自己new的东西没有进入Spring容器的门票
引入别人的答案:
即某个类中的成员,如果是采用@Autowired注入Spring Bean,则当前类的实例,必须也是Spring Bean才能成功注入,即该实例不能用new xxx()来获得,这种方式获得的实例无法调用@Autowired注入的Bean,应该也采用@Autowired注入
解决方案
所以,解决方案很简单
public class TestController{
@Autowired
Test test;
@GetMapper("...省略")
public void testMethod(){
this.test.send();
}
}