关于@Autowired注解和静态方法及new的关系

目录

  • @Autowired注解和静态方法及new关系
    • 一、@autowired 与new
    • 二、@autowired 与静态方法
      • 1.spring框架应用中有些静态方法需要依赖被容器管理的类
      • 2.原理剖析
      • 3.解决办法
  • @Autowired和new对象有什么区别
    • 如下例子所示
      • 小结

      @Autowired注解和静态方法及new关系
      一、@autowired 与new
      new出来的对象无法调用@Autowired注入的Spring Bean,否则报空指针异常,
      @Autowired注入Spring Bean,则当前类必须也是Spring Bean才能调用它,不能用new xxx()来获得对象,这种方式获得的对象无法调用@Autowired注入的Bean。

      二、@autowired 与静态方法

      1.spring框架应用中有些静态方法需要依赖被容器管理的类 像这样
      public class CeErrorAlarm{undefined@autowiredprivate static CeAlarmDao ceAlarmDao; public static ceErrorAlarm(){undefinedceAlarmDao.insert(); }}

      这样一定会报java.lang.NullPointerException: null异常。

      2.原理剖析 静态变量、类变量不是对象属性,而是一个类的属性,所以静态方法是属于类(class)的,普通方法才是属于实体对象的(new出来的对象),spring注入是在容器中实例化对象,所以不能使用静态方法。
      而使用静态变量、类变量扩大了静态方法的使用范围。
      静态方法在spring中是不推荐使用的,依赖注入的主要目的是让容器去产生一个对象的实例,然后交给spring容器管理,在整个生命周期中使用他们,更加方便灵活
      一旦你使用静态方法,就不再需要去产生这个类的实例,这会让testing变得更加困难,同时你也不能为一个给定的类,依靠注入方式去产生多个具有不同的依赖环境的实例,这种static field是隐含共享的,并且是一种global全局状态,spring同样不推荐这样去做。

      3.解决办法 1.将@autowired注解加在构造方法上
      public class CeErrorAlarm{undefinedprivate static CeAlarmDao ceAlarmDao; @autowiredpublic CeErrorAlarm(CeAlarmDao ceAlarmDao){undefinedCeErrorAlarm.ceAlarmDao=ceAlarmDao; //将利用构造方法自动注入的对象赋值给static}public static ceErrorAlarm(){undefinedceAlarmDao.insert(); }}

      2.用@PostConstruct注解
      public class CeErrorAlarm{undefinedprivate static CeAlarmDao ceAlarmDao; @AutowiredprivateCeAlarmDao ceAlarmDao2; @PostConstructpublic void init() {undefinedceAlarmDao=ceAlarmDao2; //原理类似}public static ceErrorAlarm(){undefinedceAlarmDao.insert(); }}

      @PostConstruct:被@PostConstruct修饰的方法会在服务器加载Servle的时候运行,并且只会被服务器执行一次。PostConstruct在构造函数之后执行,init()方法之前执行。(PreDestroy()方法在destroy()方法执行执行之后执行)

      @Autowired和new对象有什么区别 为什么在new 对象里面使用自动注入对象会报空指针异常?
      根本原因在于当Spring框架帮我们管理的时候就会自动的初始化接下来会用到的属性,而通过new对象的方式,在该new对象中使用到的一些实例就需要自己去做初始化,否则就会报空指针异常。

      如下例子所示
      TestService 通过@Autowired注入,那么Spring容器就会自动注入TestService 中会用到的TestDao。如例一所示。
      例一:
      @RestController@RequestMapping(value = "https://www.it610.com/test")public class TestController {@Autowiredprivate TestService testService; @RequestMapping(value = "https://www.it610.com/print",method = RequestMethod.GET)public void test() {testService.test(); }}@Servicepublic class TestService {@Autowiredprivate TestDao testDao; public void test() {testDao.test(); }}

      如果TestService 通过new对象方式新建的话,Spring容器就不会自动注入TestDao,此时testDao为null,会报空指针异常。此时就需要在TestService中自己new一个TestDao对象。如例二所示。
      例二:
      @RestController@RequestMapping(value = "https://www.it610.com/test")public class TestController {private TestService testService = new TestService (); @RequestMapping(value = "https://www.it610.com/print",method = RequestMethod.GET)public void test() {testService.test(); }}@Servicepublic class TestService {@Autowiredprivate TestDao testDao; public void test() {TestDaotestDao = new TestDao (); testDao.test(); }}


      小结
      在程序启动时,Spring会按照一定的加载链来加载并初始化Spring容器中的组件。
      例如:在A中注入B,B中注入C。在A中调用B,来使用B中调用C的方法时,如果不采用自动注入,而是使用new对象方式的话,就会报空指针异常(因为B中的C并没有被初始化)。
      【关于@Autowired注解和静态方法及new的关系】以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

        推荐阅读