问题描述:
最近写一个定时任务类,想要在controller中去调用这个类进行测试,采用了new对象的方式,结果程序运行到这个定时任务类中注入的service的调用方法这行报空指针异常,将new对象的方式改为在controller中@Autowird,再去调用这个类中注入的service的方法时就可以正常运行了
前提是定时任务类上加了@Component注解,这样才能被类路径扫描来自动侦测以及自动装配到Spring容器中
理解:@Component是声明可用,@Autowird是正式使用
原因分析:
这里需要理解Java中new对象和@Autowird注入对象的本质和区别
- new的对象是不能调用这个类对象中注入的service的,因为这个service类是被依赖注入的,自己new的对象脱离了Spring的管理,是不会被容器注入的,即手动new的对象,内部用@Autowird注解是无效的,为null。若让该autowire生效,则需要通过容器来管理对象
- 而使用@Autowird,相当于setter,在对象注入之前已经实例化了,是在这个接口注解的时候实例化的,Spring初始化了可以提供给这个类使用所需要的其他环境变量(如类中注入的这个service),即注入类需要某个对象时,容器会自动注入
解决方案:
- 方案一:在controller中@Autowird这个定时任务类,可以正常使用注入的service
- 方案二:保持new的方式使用这个定时任务类,在new的时候可以通过参数的方式将这个service传递进去,前提是service作为定时任务类的一个成员属性,在这个定时任务类中添加无参和有参构造(参数为目标service)
拓展知识点:由于Spring默认创建的对象都是单例的,在类上加
@Scope(“prototype”)
这样就可以在不同的线程中创建不同的实例
注意:如果是在一个单例类A中声明一个非单例的成员类B,是无效的,除非能再次创建这个单例类A的实例,才能创建出另一个类B的实例对象