【Spring错误笔记】自己new出来的bean中被@Autowired注解修饰的属性报空指针异常:java.lang.NullPointException

自己new出来的bean中被@Autowired注解修饰的属性报空指针异常


异常描述

原本我再测试RabbitMQ的发送程序,里面用到了一个AmqpTemplate接口,用了@Autowired注解。但是当我使用AmqpTemplateconvertAndSend()方法时却总报空指针异常 :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();
	}
}

你可能感兴趣的:(日常错误)