轉載:https://www.iteye.com/blog/blessht-1162131
平常的java開發中,程序員在某個類中需要依賴其它類的方法,則通常是new一個依賴類再調用類實例的方法,這種開發存在的問題是new的類實例不好統一管理,spring提出了依賴注入的思想,即依賴類不由程序員實例化,而是通過spring容器幫我們new指定實例並且將實例注入到需要該對象的類中。依賴注入的另一種說法是“控制反轉”,通俗的理解是:平常我們new一個實例,這個實例的控制權是我們程序員,而控制反轉是指new實例工作不由我們程序員來做而是交給spring容器來做。
spring有多種依賴注入的形式,下面僅介紹spring通過xml進行IOC配置的方式:
- Set注入
這是最簡單的注入方式,假設有一個SpringAction,類中需要實例化一個SpringDao對象,那么就可以定義一個private的SpringDao成員變量,然后創建SpringDao的set方法(這是ioc的注入入口):
1 package com.bless.springdemo.action; 2 public class SpringAction { 3 //注入對象springDao 4 private SpringDao springDao; 5 //一定要寫被注入對象的set方法 6 public void setSpringDao(SpringDao springDao) { 7 this.springDao = springDao; 8 } 9 10 public void ok(){ 11 springDao.ok(); 12 } 13 }
隨后編寫spring的xml文件,<bean>中的name屬性是class屬性的一個別名,class屬性指類的全名,因為在SpringAction中有一個公共屬性Springdao,所以要在<bean>標簽中創建一個<property>標簽指定SpringDao。<property>標簽中的name就是SpringAction類中的SpringDao屬性名,ref指下面<bean name="springDao"...>,這樣其實是spring將SpringDaoImpl對象實例化並且調用SpringAction的setSpringDao方法將SpringDao注入:
- 構造器注入
這種方式的注入是指帶有參數的構造函數注入,看下面的例子,我創建了兩個成員變量SpringDao和User,但是並未設置對象的set方法,所以就不能支持第一種注入方式,這里的注入方式是在SpringAction的構造函數中注入,也就是說在創建SpringAction對象時要將SpringDao和User兩個參數值傳進來:
在XML文件中同樣不用<property>的形式,而是使用<constructor-arg>標簽,ref屬性同樣指向其它<bean>標簽的name屬性:
下面是設置index,就是參數位置:
另一種是設置參數類型:
- 靜態工廠的方法注入
靜態工廠顧名思義,就是通過調用靜態工廠的方法來獲取自己需要的對象,為了讓spring管理所有對象,我們不能直接通過"工程類.靜態方法()"來獲取對象,而是依然通過spring注入的形式獲取:
同樣看關鍵類,這里我需要注入一個FactoryDao對象,這里看起來跟第一種注入一模一樣,但是看隨后的xml會發現有很大差別:
Spring的IOC配置文件,注意看<bean name="staticFactoryDao">指向的class並不是FactoryDao的實現類,而是指向靜態工廠DaoFactory,並且配置 factory-method="getStaticFactoryDaoImpl"指定調用哪個工廠方法:
- 實例工廠的方法注入
實例工廠的意思是獲取對象實例的方法不是靜態的,所以你需要首先new工廠類,再調用普通的實例方法:
那么下面這個類沒什么說的,跟前面也很相似,但是我們需要通過實例工廠類創建FactoryDao對象:
1 public class SpringAction { 2 //注入對象 3 private FactoryDao factoryDao; 4 public void factoryOk() { 5 factoryDao.saveFactory(); 6 } 7 public void setFactoryDao(FactoryDao factoryDao) { 8 this.factoryDao = factoryDao; 9 } 10 }
最后看spring配置文件:
- 總結
Spring IOC注入方式用得最多的是(1)(2)種,多謝多練就會非常熟練。
另外注意:通過Spring創建的對象默認是單例的,如果需要創建多實例對象可以在<bean>標簽后面添加一個屬性: