IOC,Inversion of Control,控制反轉,是面向對象編程的一種設計思想。html
原文地址:http://jinnianshilongnian.iteye.com/blog/1413846web
1.二、IoC是什麼面試
Ioc,在Java開發中,Ioc意味着將你設計好的對象交給容器控制,而不是傳統的在你的對象內部直接控制。如何理解好Ioc呢?理解好Ioc的關鍵是要明確「誰控制誰,控制什麼,爲什麼是反轉(有反轉就應該有正轉了),哪些方面反轉了」,那咱們來深刻分析一下:spring
●誰控制誰,控制什麼:傳統Java SE程序設計,咱們直接在對象內部經過new進行建立對象,是程序主動去建立依賴對象;而IoC是有專門一個容器來建立這些對象,即由Ioc容器來控制對 象的建立;誰控制誰?固然是IoC 容器控制了對象;控制什麼?那就是主要控制了外部資源獲取(不僅是對象包括好比文件等)。編程
●爲什麼是反轉,哪些方面反轉了:有反轉就有正轉,傳統應用程序是由咱們本身在對象中主動控制去直接獲取依賴對象,也就是正轉;而反轉則是由容器來幫忙建立及注入依賴對象;爲什麼是反轉?由於由容器幫咱們查找及注入依賴對象,對象只是被動的接受依賴對象,因此是反轉;哪些方面反轉了?依賴對象的獲取被反轉了。數組
圖1-2有IoC/DI容器後程序結構示意圖session
1.二、IoC能作什麼app
IoC 不是一種技術,只是一種思想,一個重要的面向對象編程的法則,它能指導咱們如何設計出鬆耦合、更優良的程序。傳統應用程序都是由咱們在類內部主動建立依賴對象,從而致使類與類之間高耦合,難於測試;有了IoC容器後,把建立和查找依賴對象的控制權交給了容器,由容器進行注入組合對象,因此對象與對象之間是 鬆散耦合,這樣也方便測試,利於功能複用,更重要的是使得程序的整個體系結構變得很是靈活。框架
其實IoC對編程帶來的最大改變不是從代碼上,而是從思想上,發生了「主從換位」的變化。應用程序本來是老大,要獲取什麼資源都是主動出擊,可是在IoC/DI思想中,應用程序就變成被動的了,被動的等待IoC容器來建立並注入它所須要的資源了。post
IoC很好的體現了面向對象設計法則之一—— 好萊塢法則:「別找咱們,咱們找你」;即由IoC容器幫對象找相應的依賴對象並注入,而不是由對象主動去找。
1.三、IoC和DI
DI—Dependency Injection,即「依賴注入」:組件之間依賴關係由容器在運行期決定,形象的說,即由容器動態的將某個依賴關係注入到組件之中。依賴注入的目的並不是爲軟件系統帶來更多功能,而是爲了提高組件重用的頻率,併爲系統搭建一個靈活、可擴展的平臺。經過依賴注入機制,咱們只須要經過簡單的配置,而無需任何代碼就可指定目標須要的資源,完成自身的業務邏輯,而不須要關心具體的資源來自何處,由誰實現。
理解DI的關鍵是:「誰依賴誰,爲何須要依賴,誰注入誰,注入了什麼」,那咱們來深刻分析一下:
●誰依賴於誰:固然是應用程序依賴於IoC容器;
●爲何須要依賴:應用程序須要IoC容器來提供對象須要的外部資源;
●誰注入誰:很明顯是IoC容器注入應用程序某個對象,應用程序依賴的對象;
●注入了什麼:就是注入某個對象所須要的外部資源(包括對象、資源、常量數據)。
IoC和DI由什麼關係呢?其實它們是同一個概念的不一樣角度描述,因爲控制反轉概念比較含糊(可能只是理解爲容器控制對象這一個層面,很難讓人想到誰來維護對象關係),因此2004年大師級人物Martin Fowler又給出了一個新的名字:「依賴注入」,相對IoC 而言,「依賴注入」明確描述了「被注入對象依賴IoC容器配置依賴對象」。
看過不少對Spring的Ioc理解的文章,好多人對Ioc和DI的解釋都晦澀難懂,反正就是一種說不清,道不明的感受,讀完以後依然是一頭霧水,感受就是開濤這位技術牛人寫得特別通俗易懂,他清楚地解釋了IoC(控制反轉) 和DI(依賴注入)中的每個字,讀完以後給人一種豁然開朗的感受。我相信對於初學Spring框架的人對Ioc的理解應該是有很大幫助的。
面向接口編程的好處:方便程序的擴展(好比集合),多態
好的程序設計知足OCP原則,在儘可能不修改程序源代碼的基礎上進行擴展,
工廠模式:工廠+反射+配置文件實現解耦和
1 將實現類交給Spring管理
1 <?xml version="1.0" encoding="UTF-8"?> 2 <beans xmlns="http://www.springframework.org/schema/beans" 3 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" 4 xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> 5 <bean id="UserService" class="com.itheima.demo1.UserServiceImpl"> 6 <property name="name" value="花花" /> 7 </bean> 8 </beans>
2 編寫測試類
/** * src目錄下的 */ @Test public void demo2(){ ApplicationContext applicationContext=new ClassPathXmlApplicationContext("applicationContext.xml"); UserService userService = (UserService) applicationContext.getBean("UserService"); userService.save(); }
配置文件能夠放在任意盤符下
/** * 配置文件放在D盤目錄下 */ @Test public void demo3(){ ApplicationContext applicationContext=new FileSystemXmlApplicationContext("D:\\applicationContext.xml"); UserService userService = (UserService) applicationContext.getBean("UserService"); userService.save(); }
IOC:控制反轉,將建立對象權交給spring管理
DI:依賴注入,前提必須有IOC的環境,Spring 管理的這個類的時候將類的屬性注入進來
面向對象的三種關係
(1)依賴
class A(){}
Class B(){public void xxx(A a){}}咱們這時候說B依賴A
(2)繼承
(3)聚合
Spring工廠類
ApplicationCotext類的UML類圖
id :使用了約束中的惟一約束
name :沒有使用約束中的惟一約束(理論上能夠出現重複,可是在實際的開發中不能重複)
scope :Bean的做用範圍
singleton :默認的,Spring會採用單例模式建立這個對象。
prototype :多例模式。(Struts2和Spring整合必定會用到)
request :應用在web項目中,Spring建立這個類之後,將這個類存入到request範圍中。
session :應用在web項目中,Spring建立這個類之後,將這個類存入到session範圍中。
globalsession :應用在web項目中,必須在porlet環境下使用。可是若是沒有這種環境,相對於session。
(1)使用構造方法注入
<!--spring屬性注入的方式 --> <!--使用構造方法--> <bean id="car" class="com.itheima.demo3.Car"> <constructor-arg name="name" value="瑪莎拉蒂"/> <constructor-arg name="price" value="2000000"/> </bean>
(2)使用set方法注入普通類型
<!--使用set方法注入普通類型--> <bean id="car2" class="com.itheima.demo3.Car2"> <property name="name" value="一汽大衆"/> <property name="price" value="200000"/> </bean>
使用set方法注入對象
<!--使用set方法注入對象--> <bean name="employee" class="com.itheima.demo3.Employee"> <property name="name" value="花花"/> <!--ref寫對象的id或者name--> <property name="car2" ref="car2"/> </bean>
(3)使用p名稱空間的方式(這點IDEA作的就比較好,不用去其餘文件去找,直接快速修復)
<bean id="car2" class="com.itheima.demo3.Car2" p:name="奔馳" p:price="500000"/> <bean id="employee" class="com.itheima.demo3.Employee" p:name="大花花" p:car2-ref="car2"> </bean>
(4)使用SPEL表達式注入
<!--改用spel表達式注入--> <bean id="car2" class="com.itheima.demo3.Car2" > <property name="name" value="#{carInfo.name}"/> <property name="price" value="#{carInfo.jisuanprice()}"/> </bean> <bean id="employee" class="com.itheima.demo3.Employee"> <property name="name" value="#{'大花'}"/> <property name="car2" value="#{car2}" /><!--用spel不用ref,都用value--> </bean> <import resource="applicationContext2.xml"/>
擴展:集合類型屬性的注入
1 <?xml version="1.0" encoding="UTF-8"?> 2 <beans xmlns="http://www.springframework.org/schema/beans" 3 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" 4 xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> 5 <!--注入數組--> 6 <bean id="Collection" class="com.itheima.demo4.Collection"> 7 <property name="arrs" > 8 <list> 9 <value>楊冪</value> 10 <value>楊紫</value> 11 <value>花花</value> 12 <value>李四</value> 13 </list> 14 </property> 15 <!--注入list集合--> 16 <property name="list" > 17 18 <list> 19 <value>張三</value> 20 <value>李四</value> 21 <value>王五</value> 22 </list> 23 </property> 24 <!--注入set集合--> 25 <property name="set"> 26 27 <set> 28 <value>aaa</value> 29 <value>bbb</value> 30 <value>xxx</value> 31 </set> 32 </property> 33 <!--注入Map集合--> 34 <property name="map"> 35 36 <map> 37 <entry key="aa" value="11"/> 38 <entry key="bb" value="22"/> 39 <entry key="cc" value="33"/> 40 </map> 41 </property> 42 </bean> 43 </beans>
養成良好的代碼註釋習慣,方便後期的查看。
往期內容: