详解 IOC
什么是IOC:
IOC—Inversion Of Control,即“控制反轉(zhuǎn)”,不是什么技術(shù),而是一種設(shè)計(jì)思想。在Java開發(fā)中,IOC意味著將你設(shè)計(jì)好的對象交給容器控制,而不是傳統(tǒng)的在你的對象內(nèi)部直接控制如何理解好IOC呢?理解好IOC的關(guān)鍵是要明確“誰控制誰,控制什么,為何是反轉(zhuǎn)(有反轉(zhuǎn)就應(yīng)該有正轉(zhuǎn)了),哪些方面反轉(zhuǎn)了”,那我們來深入分析一下:
誰控制誰,控制什么:傳統(tǒng)Java SE程序設(shè)計(jì),我們直接在對象內(nèi)部通過new進(jìn)行創(chuàng)建對象,是程序主動去創(chuàng)建依賴對象;而IOC是有專門一個容器來創(chuàng)建這些對象,即由IOC容器來控制對象的創(chuàng)建;誰控制誰?當(dāng)然是IOC 容器控制了對象;控制什么?那就是主要控制了外部資源獲取(不只是對象包括比如文件等)。
為何是反轉(zhuǎn),哪些方面反轉(zhuǎn)了:有反轉(zhuǎn)就有正轉(zhuǎn),傳統(tǒng)應(yīng)用程序是由我們自己在對象中主動控制去直接獲取依賴對象,也就是正轉(zhuǎn);而反轉(zhuǎn)則是由容器來幫忙創(chuàng)建及注入依賴對象;為何是反轉(zhuǎn)?因?yàn)橛扇萜鲙臀覀儾檎壹白⑷胍蕾噷ο?#xff0c;對象只是被動的接受依賴對象,所以是反轉(zhuǎn);哪些方面反轉(zhuǎn)了?依賴對象的獲取被反轉(zhuǎn)了。
IOC能做什么:
IOC不是一種技術(shù),只是一種思想,一個重要的面向?qū)ο缶幊痰姆▌t,它能指導(dǎo)我們?nèi)绾卧O(shè)計(jì)出松耦合、更優(yōu)良的程序。傳統(tǒng)應(yīng)用程序都是由我們在類內(nèi)部主動創(chuàng)建依賴對象,從而導(dǎo)致類與類之間高耦合,難于測試;有了IOC容器后,把創(chuàng)建和查找依賴對象的控制權(quán)交給了容器,由容器進(jìn)行注入組合對象,所以對象與對象之間是松散耦合,這樣也方便測試,利于功能復(fù)用,更重要的是使得程序的整個體系結(jié)構(gòu)變得非常靈活。
其實(shí)IOC對編程帶來的最大改變不是從代碼上,而是從思想上,發(fā)生了“主從換位”的變化。應(yīng)用程序原本是老大,要獲取什么資源都是主動出擊,但是在IOC/DI思想中,應(yīng)用程序就變成被動的了,被動的等待IOC容器來創(chuàng)建并注入它所需要的資源了。
IOC很好的體現(xiàn)了面向?qū)ο笤O(shè)計(jì)法則之一—— 好萊塢法則:“別找我們,我們找你”;即由IOC容器幫對象找相應(yīng)的依賴對象并注入,而不是由對象主動去找。
?
IOC和DI的關(guān)系:
DI—Dependency Injection,即“依賴注入”:是組件之間依賴關(guān)系由容器在運(yùn)行期決定,形象的說,即由容器動態(tài)的將某個依賴關(guān)系注入到組件之中。依賴注入的目的并非為軟件系統(tǒng)帶來更多功能,而是為了提升組件重用的頻率,并為系統(tǒng)搭建一個靈活、可擴(kuò)展的平臺。通過依賴注入機(jī)制,我們只需要通過簡單的配置,而無需任何代碼就可指定目標(biāo)需要的資源,完成自身的業(yè)務(wù)邏輯,而不需要關(guān)心具體的資源來自何處,由誰實(shí)現(xiàn)。
理解DI的關(guān)鍵是:“誰依賴誰,為什么需要依賴,誰注入誰,注入了什么”,那我們來深入分析一下:
誰依賴于誰:當(dāng)然是應(yīng)用程序依賴于IOC容器;
為什么需要依賴:應(yīng)用程序需要IOC容器來提供對象需要的外部資源;
誰注入誰:很明顯是IOC容器注入應(yīng)用程序某個對象,應(yīng)用程序依賴的對象;
注入了什么:就是注入某個對象所需要的外部資源(包括對象、資源、常量數(shù)據(jù))。
IOC和DI由什么關(guān)系呢?其實(shí)它們是同一個概念的不同角度描述,由于控制反轉(zhuǎn)概念比較含糊(可能只是理解為容器控制對象這一個層面,很難讓人想到誰來維護(hù)對象關(guān)系),所以2004年大師級人物Martin Fowler又給出了一個新的名字:“依賴注入”,相對IOC 而言,“依賴注入”明確描述了“被注入對象依賴IOC容器配置依賴對象”。
?
IOC和DI的區(qū)別
1、IOC:控制反轉(zhuǎn),把對象創(chuàng)建交給spring進(jìn)行配置。
2、DI:依賴注入,向類里面的屬性設(shè)置初始值。
3、關(guān)系:依賴注入不能單獨(dú)存在,需要在IOC基礎(chǔ)之上完成操作。
?
IOC底層使用技術(shù)
(1)xml配置文件?
(2)dom4j解析xml
(3)工廠設(shè)計(jì)模式
(4)反射
?
我們通過創(chuàng)建對象的三種方式來蘭IOC的來歷:
我們在UserServlet中引用UserService的對象
方案一:直接在 UserServlet中 new 出 UserService的對象。(左圖)
方案二:我們使用工廠模式,調(diào)用工廠類中的靜態(tài)方法getService()獲取UserService對象? (右圖)
?
方案三:我們使用 Spring,而Spring就是通過?xml配置文件 ——dom4j解析xml——通過反射創(chuàng)建對象。
使用SpringIOC創(chuàng)建對象我們可以不用關(guān)心對象所在的包(對象從哪里來),從而做到松耦合。
??
?
使用SpringIOC
?
使用Spring最基本的功能的時候,我們只需要關(guān)注上圖中的Core Container(核心容器)
Beans、Core、Context、SpEL 并完成以下步驟:
1、引入核心容器的相關(guān)4個jar包(引入依賴)
2、引入支持日志輸出的jar包和Junit單元測試(引入依賴)
??
?
3、創(chuàng)建Spring配置文件,創(chuàng)建配置類 applicationContext.xml
(1)建議放到 src 下面,官方建議 applicationContext.xml
(2)引入 schema 約束 【了解更多關(guān)于schema?點(diǎn)我】
(3)配置對象的創(chuàng)建(我們以配置實(shí)體類User為例)
?
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"><!-- ioc入門 --><bean id="user" class="com.bjxb.ioc.User"></bean> </beans>?
4、測試代碼(maven項(xiàng)目)applicationContext.xml在resource下的spring文件夾中
?
public class TestIOC {@Testpublic void testUser() {//1.加載Spring配置文件,創(chuàng)建對象ApplicationContext context = new ClassPathXmlApplicationContext("/spring/applicationContext.xml");//2.得到配置創(chuàng)建的對象User user = (User) context.getBean("user");System.out.println(user);} }?
?
?
?
?
注:如果想要更加深入的了解IOC和DI,請參考大師級人物Martin Fowler的一篇經(jīng)典文章
《Inversion of Control Containers and the Dependency Injection pattern》,原文地址:http://www.martinfowler.com/articles/injection.html。
方案二:我們使用工廠模式,調(diào)用工廠類中的靜態(tài)方法getService()獲取UserService對象
轉(zhuǎn)載于:https://www.cnblogs.com/xb1223/p/10148503.html
總結(jié)