很多人認爲IOC就是DI他們之間是對等的,其實不然。
很多文章中都提到他們的分別依賴:
誰依賴於誰?爲什麼需要依賴?
注入:誰注入於誰?到底注入什麼?
控制反轉:誰控制誰?控制什麼?
這種描述很難讓別人理解,感覺有點教科書的意味。下面我就簡單的說下我的看法:
一、IOC和DI區別
IOC( inversion of control)控制反轉,控制反轉是一種思想,結合程序通俗的講,創建對象的方式反轉了。
以前對象的創建是由我們開發人員自己維護創建,包括依賴關係都是自己進行注入(創建依賴對象new),例如:我們經常提到的三層框架Web層、Service層、Dao層。web層依賴於service層,service依賴於dao層,以前我們管理service層調用dao層的數據時,都會new一個對象調用其中的方法。
使用spring後對象的創建及依賴的對象都由spring完成創建和注入,控制反轉就是對象的創建方式,從我們自己創建到交個spring(程序)進行創建管理
DI(Dependency Injection)依賴注入,依賴注入是一種技術,它是對控制反轉這種思想進行技術支撐。依賴注入即控制反轉中舉的例子。
二、Spring創建Bean對象的三種方式
首先我們看看不用spring時如何創建對象
public class UserServiceImpl {
public List<User> getUserList(){
//servi層調用dao層的方法,需要手動創建dao對象
UserDao userDao = new UserDao();
return userDao.getUserList();
}
}
使用spring,spring就會自動創建對象(前提按照一定的規則,這個規則是spring規定的)
第一種方式構造器注入創建(默認的創建方式)
public class User {
public User() {
System.out.println("User對象構造器注入創建!!!!");
}
private String name;
private Integer age;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
@Override
public String toString() {
return "User [name=" + name + ", age=" + age + "]";
}
}
Spring配置文件applicationContext.xml中配置
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns="http://www.springframework.org/schema/beans"
xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd ">
<!-- set方式注入: -->
<bean name="user" class="cn.mingxungu.bean.User" />
</beans>
測試代碼
@Test
public void fun1(){
//1 創建容器對象
ApplicationContext ac = new ClassPathXmlApplicationContext("cn/mingxungu/c_injection/applicationContext.xml");
//2 向容器"要"user對象
User u = (User) ac.getBean("user");
//3 打印user對象
System.out.println(u);
}
結果:
User對象構造器注入創建!!!!
User [name=null, age=null]
第二種方式靜態工廠創建(瞭解)
public class UserFactory {
public static User createUser(){
System.out.println("靜態工廠創建User");
return new User();
}
}
Spring配置文件applicationContext.xml中配置
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns="http://www.springframework.org/schema/beans"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd ">
<!-- 創建方式2:靜態工廠創建
調用UserFactory的createUser方法創建名爲user2的對象.放入容器
-->
<bean name="user2"
class="cn.mingxungu.b_create.UserFactory"
factory-method="createUser" ></bean>
</beans>
測試代碼
//創建方式2:靜態工廠
@Test
public void fun2(){
//1 創建容器對象
ApplicationContext ac = new ClassPathXmlApplicationContext("cn/mingxungu/b_create/applicationContext.xml");
//2 向容器"要"user對象
User u = (User) ac.getBean("user2");
//3 打印user對象
System.out.println(u);
}
輸出結果
靜態工廠創建User
User [name=null, age=null]
注意:靜態工廠方法在配置文件中制定了工廠方法名稱factory-method="createUser"
第三種方式實例工廠創建(瞭解)
public class UserFactory {
public User createUser2(){
System.out.println("實例工廠創建User");
return new User();
}
}
Spring配置文件applicationContext.xml中配置
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns="http://www.springframework.org/schema/beans"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd ">
<!-- 創建方式3:實例工廠創建
調用UserFactory對象的createUser2方法創建名爲user3的對象.放入容器
-->
<bean name="user3"
factory-bean="userFactory"
factory-method="createUser2" ></bean>
<bean name="userFactory"
class="cn.mingxungu.b_create.UserFactory" ></bean>
</beans>
測試代碼
//創建方式3:實例工廠
@Test
public void fun3(){
//1 創建容器對象
ApplicationContext ac = new ClassPathXmlApplicationContext("cn/mingxungu/b_create/applicationContext.xml");
//2 向容器"要"user對象
User u = (User) ac.getBean("user3");
//3 打印user對象
System.out.println(u);
}
輸出結果
實例工廠創建User
User [name=null, age=null]
總結:
一、<bean>元素指定了factory-method屬性,Spring就不再調用構造器來創建Bean實例,而是調用工廠方法來創建Bean實例。如果同時指定了class和factory-method兩個屬性,Spring就會調用靜態工廠方法來創建Bean。Spring將先解析配置文件,並根據配置文件指定的信息,通過反射調用靜態工廠類的靜態工廠方法,將該靜態工廠方法的返回值作爲Bean實例。在這個過程中,Spring不再負責創建Bean實例,Bean實例是由用戶提供的靜態工廠類負責創建的。
二、實例工廠方法與靜態工廠方法只有一點不同:調用靜態工廠方法只需要使用工廠類即可,而調用實例工廠方法則需要工廠實例。所以在配置時,靜態工廠方法使用class指定靜態工廠類,實例工廠方法使用factory-bean指定工廠實例。
採用實例工廠方法創建Bean的<bean>元素時需要指定兩個屬性:
factory-bean:工廠bean的id
factory-method:實例工廠的工廠方法
三、Bean 生命週期
詳解:
Bean的完整生命週期經歷了各種方法調用,這些方法可以劃分爲以下幾類:
1、Bean自身的方法:這個包括了Bean本身調用的方法和通過配置文件中<bean>的init-method和destroy-method指定的方法
2、Bean級生命週期接口方法:這個包括了BeanNameAware、BeanFactoryAware、InitializingBean和DiposableBean這些接口的方法
3、容器級生命週期接口方法:這個包括了InstantiationAwareBeanPostProcessor 和 BeanPostProcessor 這兩個接口實現,一般稱它們的實現類爲“後處理器”。
4、工廠後處理器接口方法:這個包括了AspectJWeavingEnabler, ConfigurationClassPostProcessor, CustomAutowireConfigurer等等非常有用的工廠後處理器 接口的方法。工廠後處理器也是容器級的。在應用上下文裝配配置文件之後立即調用。說明:
在配置文件aplicationContext.xml中配置的bean的生命週期
<bean name="user" class="cn.mingxungu.bean.User" init-method="init" destroy-method="destory" />
init-method="init":配置一個方法作爲生命週期初始化方法,在該Bean中編寫一個init方法。spring會在對象創建之後立即調用.
destroy-method="destory":配置一個方法作爲生命週期的銷燬方法,在該Bean中編寫一個destory方法。spring容器在關閉並銷燬所有容器中該對象之前調用.
詳細的測試參考:http://www.cnblogs.com/zrtqsk/p/3735273.html
四、Bean屬性的注入
注入方式:set方法注入、構造函數注入、p名稱空間注入、spel注入
set方法注入:
<!-- set方式注入: -->
<bean name="user" class="cn.mingxungu.bean.User" >
<!--值類型注入: 爲User對象中名爲name的屬性注入tom作爲值 -->
<property name="name" value="tom" ></property>
<property name="age" value="18" ></property>
<!-- 引用類型注入: 爲car屬性注入下方配置的car對象 -->
<property name="car" ref="car" ></property>
</bean>
<!-- 將car對象配置到容器中 -->
<bean name="car" class="cn.mingxungu.bean.Car" >
<property name="name" value="蘭博基尼" ></property>
<property name="color" value="黃色" ></property>
</bean>
構造函數注入(掌握)<!-- 構造函數注入 -->
<bean name="user2" class="cn.mingxungu.bean.User" >
<!-- name屬性: 構造函數的參數名 -->
<!-- index屬性: 構造函數的參數索引 -->
<!-- type屬性: 構造函數的參數類型-->
<constructor-arg name="name" index="0" type="java.lang.Integer" value="999" ></constructor-arg>
<constructor-arg name="car" ref="car" index="1" ></constructor-arg>
</bean>
<!-- 將car對象配置到容器中 -->
<bean name="car" class="cn.mingxungu.bean.Car" >
<property name="name" value="蘭博基尼" ></property>
<property name="color" value="黃色" ></property>
</bean>
p名稱空間注入
<!-- p名稱空間注入, 走set方法
1.導入P名稱空間 xmlns:p="http://www.springframework.org/schema/p"
2.使用p:屬性完成注入
|-值類型: p:屬性名="值"
|-對象類型: p:屬性名-ref="bean名稱"
-->
<bean name="user3" class="cn.mingxungu.bean.User" p:name="jack" p:age="20" p:car-ref="car" />
<!-- 將car對象配置到容器中 -->
<bean name="car" class="cn.mingxungu.bean.Car" >
<property name="name" value="蘭博基尼" ></property>
<property name="color" value="黃色" ></property>
</bean>
spel注入
<!--
spel注入: spring Expression Language sping表達式語言
-->
<bean name="user4" class="cn.mingxungu.bean.User" >
<property name="name" value="#{user.name}" ></property>
<property name="age" value="#{user3.age}" ></property>
<property name="car" ref="car" ></property>
</bean>
<!-- 將car對象配置到容器中 -->
<bean name="car" class="cn.itcast.bean.Car" >
<property name="name" value="蘭博基尼" ></property>
<property name="color" value="黃色" ></property>
</bean>
複雜類型的注入
實體:
public class CollectionBean {
private Object[] arr;//數組類型注入
private List list;//list/set 類型注入
private Map map;//map類型注入
private Properties prop;//properties類型注入
public Object[] getArr() {
return arr;
}
public void setArr(Object[] arr) {
this.arr = arr;
}
public List getList() {
return list;
}
public void setList(List list) {
this.list = list;
}
public Map getMap() {
return map;
}
public void setMap(Map map) {
this.map = map;
}
public Properties getProp() {
return prop;
}
public void setProp(Properties prop) {
this.prop = prop;
}
@Override
public String toString() {
return "CollectionBean [arr=" + Arrays.toString(arr) + ", list=" + list + ", map=" + map + ", prop=" + prop
+ "]";
}
}
配置文件
<bean name="cb" class="cn.mingxungu.c_injection.CollectionBean" >
<!-- 如果數組中只准備註入一個值(對象),直接使用value|ref即可
<property name="arr" value="tom" ></property>
-->
<!-- array注入,多個元素注入 -->
<property name="arr">
<array>
<value>tom</value>
<value>jerry</value>
<ref bean="user4" />
</array>
</property>
<!-- 如果List中只准備註入一個值(對象),直接使用value|ref即可
<property name="list" value="jack" ></property>-->
<property name="list" >
<list>
<value>jack</value>
<value>rose</value>
<ref bean="user3" />
</list>
</property>
<!-- map類型注入 -->
<property name="map" >
<map>
<entry key="url" value="jdbc:mysql:///crm" ></entry>
<entry key="user" value-ref="user4" ></entry>
<entry key-ref="user3" value-ref="user2" ></entry>
</map>
</property>
<!-- prperties 類型注入 -->
<property name="prop" >
<props>
<prop key="driverClass">com.jdbc.mysql.Driver</prop>
<prop key="userName">root</prop>
<prop key="password">1234</prop>
</props>
</property>
</bean>
五、Bean屬性scope
scope類型:
singleton(默認值):單例對象.被標識爲單例的對象在spring容器中只會存在一個實例
prototype:多例原型.被標識爲多例的對象,每次再獲得纔會創建.每次創建都是新的對象.整合struts2時,ActionBean必須配置爲多例的.
request:web環境下.對象與request生命週期一致.
session:web環境下,對象與session生命週期一致.
說明:前兩個重點掌握,其他兩個一般用不到
singleton測試
aplicationContext.xml配置文件
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns="http://www.springframework.org/schema/beans"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd ">
<!-- 創建方式1:空參構造創建 -->
<bean name="user" class="cn.mingxungu.bean.User" scope="singleton"/>
</beans>
測試
@Test
//scope:singleton 單例
//scope:prototype 多例
public void fun4(){
//1 創建容器對象
ApplicationContext ac = new ClassPathXmlApplicationContext("cn/mingxungu/b_create/applicationContext.xml");
//2 向容器"要"user對象
User u = (User) ac.getBean("user");
User u2 = (User) ac.getBean("user");
System.out.println("比較是否是一個對象"+(u==u2)); //單例:true 多例:false
}
結果
User對象構造器注入創建!!!!
比較是否是一個對象true
prototype測試
只用修改上面的配置文件即可
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns="http://www.springframework.org/schema/beans"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd ">
<!-- 創建方式1:空參構造創建 -->
<bean name="user" class="cn.mingxungu.bean.User" scope="prototype"/>
</beans>
結果
User對象構造器注入創建!!!!
User對象構造器注入創建!!!!
比較是否是一個對象false
結論:注意到 “User對象構造器注入創建!!!!” 打印的次數不一樣,並且結果也不一樣,說明prototype確實是重新new了一遍