關(guān)于@Autowired注解和靜態(tài)方法及new的關(guān)系
@Autowired注解和靜態(tài)方法及new關(guān)系
一、@autowired 與new
new出來(lái)的對(duì)象無(wú)法調(diào)用@Autowired注入的Spring Bean,否則報(bào)空指針異常,
@Autowired注入Spring Bean,則當(dāng)前類必須也是Spring Bean才能調(diào)用它,不能用new xxx()來(lái)獲得對(duì)象,這種方式獲得的對(duì)象無(wú)法調(diào)用@Autowired注入的Bean。
二、@autowired 與靜態(tài)方法
1.spring框架應(yīng)用中有些靜態(tài)方法需要依賴被容器管理的類
像這樣
public class CeErrorAlarm{undefined ? ?@autowired ? ? private static CeAlarmDao ceAlarmDao; ? ?public static ceErrorAlarm(){undefined ? ? ? ? ? ceAlarmDao.insert(); ? ? } }
這樣一定會(huì)報(bào)java.lang.NullPointerException: null異常。
2.原理剖析
靜態(tài)變量、類變量不是對(duì)象屬性,而是一個(gè)類的屬性,所以靜態(tài)方法是屬于類(class)的,普通方法才是屬于實(shí)體對(duì)象的(new出來(lái)的對(duì)象),spring注入是在容器中實(shí)例化對(duì)象,所以不能使用靜態(tài)方法。
而使用靜態(tài)變量、類變量擴(kuò)大了靜態(tài)方法的使用范圍。
靜態(tài)方法在spring中是不推薦使用的,依賴注入的主要目的是讓容器去產(chǎn)生一個(gè)對(duì)象的實(shí)例,然后交給spring容器管理,在整個(gè)生命周期中使用他們,更加方便靈活
一旦你使用靜態(tài)方法,就不再需要去產(chǎn)生這個(gè)類的實(shí)例,這會(huì)讓testing變得更加困難,同時(shí)你也不能為一個(gè)給定的類,依靠注入方式去產(chǎn)生多個(gè)具有不同的依賴環(huán)境的實(shí)例,這種static field是隱含共享的,并且是一種global全局狀態(tài),spring同樣不推薦這樣去做。
3.解決辦法
1.將@autowired注解加在構(gòu)造方法上
public class CeErrorAlarm{undefined ? ? private static CeAlarmDao ceAlarmDao; ? ?@autowired ? ?public CeErrorAlarm(CeAlarmDao ceAlarmDao){undefined ? ? ? CeErrorAlarm.ceAlarmDao=ceAlarmDao;//將利用構(gòu)造方法自動(dòng)注入的對(duì)象賦值給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修飾的方法會(huì)在服務(wù)器加載Servle的時(shí)候運(yùn)行,并且只會(huì)被服務(wù)器執(zhí)行一次。PostConstruct在構(gòu)造函數(shù)之后執(zhí)行,init()方法之前執(zhí)行。(PreDestroy()方法在destroy()方法執(zhí)行執(zhí)行之后執(zhí)行)
@Autowired和new對(duì)象有什么區(qū)別
為什么在new 對(duì)象里面使用自動(dòng)注入對(duì)象會(huì)報(bào)空指針異常?
根本原因在于當(dāng)Spring框架幫我們管理的時(shí)候就會(huì)自動(dòng)的初始化接下來(lái)會(huì)用到的屬性,而通過(guò)new對(duì)象的方式,在該new對(duì)象中使用到的一些實(shí)例就需要自己去做初始化,否則就會(huì)報(bào)空指針異常。
如下例子所示
TestService 通過(guò)@Autowired注入,那么Spring容器就會(huì)自動(dòng)注入TestService 中會(huì)用到的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 通過(guò)new對(duì)象方式新建的話,Spring容器就不會(huì)自動(dòng)注入TestDao,此時(shí)testDao為null,會(huì)報(bào)空指針異常。此時(shí)就需要在TestService中自己new一個(gè)TestDao對(duì)象。如例二所示。
例二:
@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(); ? ? } }
小結(jié)
在程序啟動(dòng)時(shí),Spring會(huì)按照一定的加載鏈來(lái)加載并初始化Spring容器中的組件。
例如:在A中注入B,B中注入C。在A中調(diào)用B,來(lái)使用B中調(diào)用C的方法時(shí),如果不采用自動(dòng)注入,而是使用new對(duì)象方式的話,就會(huì)報(bào)空指針異常(因?yàn)锽中的C并沒(méi)有被初始化)。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
IntelliJ IDEA中折疊所有Java代碼,再也不怕大段的代碼了
今天小編就為大家分享一篇關(guān)于IntelliJ IDEA中折疊所有Java代碼,再也不怕大段的代碼了,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2018-10-10SpringBoot創(chuàng)建并簡(jiǎn)單使用的實(shí)現(xiàn)
這篇文章主要介紹了SpringBoot創(chuàng)建并簡(jiǎn)單使用的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10SpringBoot整合Shiro思路(最新超詳細(xì))
這篇文章主要介紹了SpringBoot整合Shiro思路(最新超詳細(xì)),本文內(nèi)容比較長(zhǎng),通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-03-03springboot實(shí)現(xiàn)發(fā)送短信驗(yàn)證碼的示例代碼
項(xiàng)目里面有用到用戶手機(jī)號(hào)注冊(cè)發(fā)短信功能,本文主要介紹了springboot實(shí)現(xiàn)發(fā)送短信驗(yàn)證碼的示例代碼,具有一定的參考價(jià)值,感興趣的可以了解一下2023-09-09詳解HTTP請(qǐng)求與響應(yīng)基礎(chǔ)及實(shí)例
這篇文章主要介紹了詳解HTTP請(qǐng)求與響應(yīng)基礎(chǔ)及實(shí)例的相關(guān)資料,這里對(duì)http的請(qǐng)求和響應(yīng)進(jìn)行詳細(xì)分析并附有實(shí)現(xiàn)實(shí)例,需要的朋友可以參考下2017-07-07Spring Boot開(kāi)發(fā)Web應(yīng)用詳解
這篇文章主要介紹了Spring Boot開(kāi)發(fā)Web應(yīng)用詳解,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-04-04查看jdk(java開(kāi)發(fā)工具包)安裝路徑的兩種方法
若已經(jīng)安裝好了jdk(java開(kāi)發(fā)工具包),也配置了環(huán)境變量,事后卻忘了安裝路徑在哪,如何查看jdk安裝路徑?本文給大家介紹了兩種查看jdk(java開(kāi)發(fā)工具包)安裝路徑的方法,需要的朋友可以參考下2023-12-12