new 出的对象无法使用 @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+"#############"

异常分析
需要了解的前提:

【1】构造方法和@Autowired和@PostConstruct的执行顺序是:constructor > @Autowired > @PostConstruct
【2】@PostConstruct修饰的方法会在@Autowired注入后执行

情况是这样的:

【1】我们在Test类中声明了@Component,所以项目启动时,Spring会构建一次Test类,放入Spring Bean容器
【2】我们在Test类中使用@Autowired注解的方式来注入属性,即通过Spring的方式来获得AmqpTemplate实例
【3】我们在TestController类中没有使用@Autowired的方式来获得Test实例,而是自行new
异常分析:
我们看到结果输出了4句话
 

 

constructer: amqpTemplate true+"#############"
init: amqpTemplate false+"#############"
constructer: amqpTemplate true+"#############"
send: amqpTemplate true+"#############"

 

 

【1】第一句话肯定是@Component注解的作用,即spring构建了Test类的实例,放入Spring Bean容器中,此时的amqpTemplate还未被注入,所以是null
【2】第二句是Spring容器构建的时候,触发了Init方法,此时的amqpTemplate已经经过了@Autowried注入,所以是false,已经不是null了。Spring容器中的amqpTemplate 被注入到Test实例中的amqpTemplate属性中。
【3】第三句开始就是重点了,是我们在TestController执行的new Test()方法,人为触发了Test的构造函数,不是Spring容器。此时的amqpTemplate实例,我们发现是true。不过在构造函数阶段是null也正常。
【4】第四句是紧跟构造函数和@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();
	}
}

 

 

  • 5
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值