@Autowired註解和靜態方法以及new的關係
一、@autowired 與new
new出來的物件無法呼叫@Autowired注入的Spring Bean,否則報空指標異常,
@Autowired注入Spring Bean,則當前類必須也是Spring Bean才能呼叫它,不能用new xxx()來獲得物件,這種方式獲得的物件無法呼叫@Autowired注入的Bean。
二、@autowired 與靜態方法
1.spring框架應用中有些靜態方法需要依賴被容器管理的類,像這樣
public class CeErrorAlarm{
@autowired private static CeAlarmDao ceAlarmDao;
public static ceErrorAlarm(){ ceAlarmDao.insert(); } } |
這樣一定會報java.lang.NullPointerException: null異常。
2.原理剖析
靜態變數、類變數不是物件屬性,而是一個類的屬性,所以靜態方法是屬於類(class)的,普通方法才是屬於實體物件的(new出來的物件),spring注入是在容器中例項化物件,所以不能使用靜態方法。
而使用靜態變數、類變數擴大了靜態方法的使用範圍。
靜態方法在spring中是不推薦使用的,依賴注入的主要目的是讓容器去產生一個物件的例項,然後交給spring容器管理,在整個生命週期中使用他們,更加方便靈活
一旦你使用靜態方法,就不再需要去產生這個類的例項,這會讓testing變得更加困難,同時你也不能為一個給定的類,依靠注入方式去產生多個具有不同的依賴環境的例項,這種static field是隱含共享的,並且是一種global全域性狀態,spring同樣不推薦這樣去做。
3.解決辦法
1.將@autowired註解加在構造方法上
public class CeErrorAlarm{
private static CeAlarmDao ceAlarmDao; @autowired public CeErrorAlarm(CeAlarmDao ceAlarmDao){ CeErrorAlarm.ceAlarmDao=ceAlarmDao;//將利用構造方法自動注入的物件賦值給static } public static ceErrorAlarm(){ ceAlarmDao.insert(); } } |
2.用@PostConstruct註解
public class CeErrorAlarm{
private static CeAlarmDao ceAlarmDao; @Autowired
@PostConstruct public static ceErrorAlarm(){ ceAlarmDao.insert(); } } |
@PostConstruct:被@PostConstruct修飾的方法會在伺服器載入Servle的時候執行,並且只會被伺服器執行一次。PostConstruct在建構函式之後執行,init()方法之前執行。(PreDestroy()方法在destroy()方法執行執行之後執行)