關于@Autowired注解和靜態(tài)方法及new的關系
@Autowired注解和靜態(tài)方法及new關系
一、@autowired 與new
new出來的對象無法調用@Autowired注入的Spring Bean,否則報空指針異常,
@Autowired注入Spring Bean,則當前類必須也是Spring Bean才能調用它,不能用new xxx()來獲得對象,這種方式獲得的對象無法調用@Autowired注入的Bean。
二、@autowired 與靜態(tài)方法
1.spring框架應用中有些靜態(tài)方法需要依賴被容器管理的類
像這樣
public class CeErrorAlarm{undefined
? ?@autowired
? ? private static CeAlarmDao ceAlarmDao;
? ?public static ceErrorAlarm(){undefined
? ? ? ? ? ceAlarmDao.insert();
? ? }
}這樣一定會報java.lang.NullPointerException: null異常。
2.原理剖析
靜態(tài)變量、類變量不是對象屬性,而是一個類的屬性,所以靜態(tài)方法是屬于類(class)的,普通方法才是屬于實體對象的(new出來的對象),spring注入是在容器中實例化對象,所以不能使用靜態(tài)方法。
而使用靜態(tài)變量、類變量擴大了靜態(tài)方法的使用范圍。
靜態(tài)方法在spring中是不推薦使用的,依賴注入的主要目的是讓容器去產生一個對象的實例,然后交給spring容器管理,在整個生命周期中使用他們,更加方便靈活
一旦你使用靜態(tài)方法,就不再需要去產生這個類的實例,這會讓testing變得更加困難,同時你也不能為一個給定的類,依靠注入方式去產生多個具有不同的依賴環(huán)境的實例,這種static field是隱含共享的,并且是一種global全局狀態(tài),spring同樣不推薦這樣去做。
3.解決辦法
1.將@autowired注解加在構造方法上
public class CeErrorAlarm{undefined
? ? private static CeAlarmDao ceAlarmDao;
? ?@autowired
? ?public CeErrorAlarm(CeAlarmDao ceAlarmDao){undefined
? ? ? CeErrorAlarm.ceAlarmDao=ceAlarmDao;//將利用構造方法自動注入的對象賦值給static
}
? ?public static ceErrorAlarm(){undefined
? ? ? ? ? ceAlarmDao.insert();
? ? }
}2.用@PostConstruct注解
public class CeErrorAlarm{undefined
? ? private static CeAlarmDao ceAlarmDao;
? ? @Autowired
? ? private ?CeAlarmDao ceAlarmDao2;
? ?@PostConstruct
? ? public void init() {undefined
? ? ? ? ceAlarmDao=ceAlarmDao2; //原理類似
? ? ? ??
? ? }
? ?public static ceErrorAlarm(){undefined
? ? ? ? ? ceAlarmDao.insert();
? ? }
}@PostConstruct:被@PostConstruct修飾的方法會在服務器加載Servle的時候運行,并且只會被服務器執(zhí)行一次。PostConstruct在構造函數之后執(zhí)行,init()方法之前執(zhí)行。(PreDestroy()方法在destroy()方法執(zhí)行執(zhí)行之后執(zhí)行)
@Autowired和new對象有什么區(qū)別
為什么在new 對象里面使用自動注入對象會報空指針異常?
根本原因在于當Spring框架幫我們管理的時候就會自動的初始化接下來會用到的屬性,而通過new對象的方式,在該new對象中使用到的一些實例就需要自己去做初始化,否則就會報空指針異常。
如下例子所示
TestService 通過@Autowired注入,那么Spring容器就會自動注入TestService 中會用到的TestDao。如例一所示。
例一:
@RestController
@RequestMapping(value = "/test")
public class TestController {
? ? @Autowired
? ? private TestService testService;
? ? @RequestMapping(value = "/print",method = RequestMethod.GET)
? ? public void test() {
? ? ? ? testService.test();
? ? }
}
@Service
public class TestService {
? ? @Autowired
? ? private TestDao testDao;
? ? public void test() {
? ? ? ? testDao.test();
? ? }
}如果TestService 通過new對象方式新建的話,Spring容器就不會自動注入TestDao,此時testDao為null,會報空指針異常。此時就需要在TestService中自己new一個TestDao對象。如例二所示。
例二:
@RestController
@RequestMapping(value = "/test")
public class TestController {
? ? private TestService testService = new TestService ();
? ? @RequestMapping(value = "/print",method = RequestMethod.GET)
? ? public void test() {
? ? ? ? testService.test();
? ? }
}
@Service
public class TestService {
? ? @Autowired
? ? private TestDao testDao;
? ? public void test() {
? ? ? ? TestDao ?testDao = new TestDao ();
? ? ? ? testDao.test();
? ? }
}小結
在程序啟動時,Spring會按照一定的加載鏈來加載并初始化Spring容器中的組件。
例如:在A中注入B,B中注入C。在A中調用B,來使用B中調用C的方法時,如果不采用自動注入,而是使用new對象方式的話,就會報空指針異常(因為B中的C并沒有被初始化)。
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
Mybatis使用useGeneratedKeys獲取自增主鍵
這篇文章主要為大家介紹了Mybatis使用useGeneratedKeys獲取自增主鍵示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-01-01
mysql+spring+mybatis實現數據庫讀寫分離的代碼配置
今天小編就為大家分享一篇關于mysql+spring+mybatis實現數據庫讀寫分離的代碼配置,小編覺得內容挺不錯的,現在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-03-03

