spring ioc和di

很多人認爲IOC就是DI他們之間是對等的,其實不然。

很多文章中都提到他們的分別依賴:

誰依賴於誰?爲什麼需要依賴?

注入:誰注入於誰?到底注入什麼?

控制反轉:誰控制誰?控制什麼?

這種描述很難讓別人理解,感覺有點教科書的意味。下面我就簡單的說下我的看法:

一、IOC和DI區別

IOC( inversion of control)控制反轉,控制反轉是一種思想結合程序通俗的講,創建對象的方式反轉了。

以前對象的創建是由我們開發人員自己維護創建,包括依賴關係都是自己進行注入(創建依賴對象new),例如:我們經常提到的三層框架Web層、Service層、Dao層。web層依賴於service層,service依賴於dao層,以前我們管理service層調用dao層的數據時,都會new一個對象調用其中的方法。

使用spring後對象的創建及依賴的對象都由spring完成創建和注入,控制反轉就是對象的創建方式,從我們自己創建到交個spring(程序)進行創建管理

DIDependency 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了一遍

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章