本文主要是介绍【Spring错误笔记】自己new出来的bean中被@Autowired注解修饰的属性报空指针异常:java.lang.NullPointException,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
自己new出来的bean中被@Autowired注解修饰的属性报空指针异常
异常描述
原本我再测试RabbitMQ的发送程序,里面用到了一个AmqpTemplate
接口,用了@Autowired
注解。但是当我使用AmqpTemplate
的convertAndSend()
方法时却总报空指针异常 :java.lang.NullPointException
错误代码
修改过的可重现问题代码:
@Component
public class Test
{ //修改过的重现问题代码@AutowiredAmqpTemplate 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{@AutowiredTest test;@GetMapper("...省略")public void testMethod(){this.test.send();}
}
这篇关于【Spring错误笔记】自己new出来的bean中被@Autowired注解修饰的属性报空指针异常:java.lang.NullPointException的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!