PHP依賴(lài)注入(DI)和控制反轉(zhuǎn)(IoC)詳解
首先依賴(lài)注入和控制反轉(zhuǎn)說(shuō)的是同一個(gè)東西,是一種設(shè)計(jì)模式,這種設(shè)計(jì)模式用來(lái)減少程序間的耦合,鄙人學(xué)習(xí)了一下,看TP官網(wǎng)還沒(méi)有相關(guān)的文章,就寫(xiě)下這篇拙作介紹一下這種設(shè)計(jì)模式,希望能為T(mén)P社區(qū)貢獻(xiàn)一些力量。
首先先別追究這個(gè)設(shè)計(jì)模式的定義,否則你一定會(huì)被說(shuō)的云里霧里,筆者就是深受其害,百度了N多文章,都是從理論角度來(lái)描述,充斥著大量的生澀詞匯,要么就是java代碼描述的,也生澀。
不管怎么樣,總算弄清楚一些了,下面就以php的角度來(lái)描述一下依賴(lài)注入這個(gè)概念。
先假設(shè)我們這里有一個(gè)類(lèi),類(lèi)里面需要用到數(shù)據(jù)庫(kù)連接,按照最最原始的辦法,我們可能是這樣寫(xiě)這個(gè)類(lèi)的:
class example { private $_db; function __construct(){ include "./Lib/Db.php"; $this->_db = new Db("localhost","root","123456","test"); } function getList(){ $this->_db->query("......");//這里具體sql語(yǔ)句就省略不寫(xiě)了 } }
過(guò)程:
在構(gòu)造函數(shù)里先將數(shù)據(jù)庫(kù)類(lèi)文件include進(jìn)來(lái);
然后又通過(guò)new Db并傳入數(shù)據(jù)庫(kù)連接信息實(shí)例化db類(lèi);
之后getList方法就可以通過(guò)$this->_db來(lái)調(diào)用數(shù)據(jù)庫(kù)類(lèi),實(shí)現(xiàn)數(shù)據(jù)庫(kù)操作。
看上去我們實(shí)現(xiàn)了想要的功能,但是這是一個(gè)噩夢(mèng)的開(kāi)始,以后example1,example2,example3....越來(lái)越多的類(lèi)需要用到db組件,如果都這么寫(xiě)的話,萬(wàn)一有一天數(shù)據(jù)庫(kù)密碼改了或者db類(lèi)發(fā)生變化了,豈不是要回頭修改所有類(lèi)文件?
ok,為了解決這個(gè)問(wèn)題,工廠模式出現(xiàn)了,我們創(chuàng)建了一個(gè)Factory方法,并通過(guò)Factory::getDb()方法來(lái)獲得db組件的實(shí)例:
class Factory { public static function getDb(){ include "./Lib/Db.php"; return new Db("localhost","root","123456","test"); } }
sample類(lèi)變成:
class example { private $_db; function __construct(){ $this->_db = Factory::getDb(); } function getList(){ $this->_db->query("......");//這里具體sql語(yǔ)句就省略不寫(xiě)了 } }
這樣就完美了嗎?再次想想一下以后example1,example2,example3....所有的類(lèi),你都需要在構(gòu)造函數(shù)里通過(guò)Factory::getDb();獲的一個(gè)Db實(shí)例,實(shí)際上由原來(lái)的直接與Db類(lèi)的耦合變?yōu)榱撕虵actory工廠類(lèi)的耦合,工廠類(lèi)只是幫你把數(shù)據(jù)庫(kù)連接信息給包裝起來(lái)了,雖然當(dāng)數(shù)據(jù)庫(kù)信息發(fā)生變化時(shí)只要修改Factory::getDb()方法就可以了,但是突然有一天工廠方法需要改名,或者getDb方法需要改名,你又怎么辦?當(dāng)然這種需求、、但有時(shí)候確實(shí)存在這種情況,一種解決方式是:
我們不從example類(lèi)內(nèi)部實(shí)例化Db組件,我們依靠從外部的注入,什么意思呢?看下面的例子:
class example { private $_db; function getList(){ $this->_db->query("......");//這里具體sql語(yǔ)句就省略不寫(xiě)了 } //從外部注入db連接 function setDb($connection){ $this->_db = $connection; } } //調(diào)用 $example = new example(); $example->setDb(Factory::getDb());//注入db連接 $example->getList();
這樣一來(lái),example類(lèi)完全與外部類(lèi)解除耦合了,你可以看到Db類(lèi)里面已經(jīng)沒(méi)有工廠方法或Db類(lèi)的身影了。我們通過(guò)從外部調(diào)用example類(lèi)的setDb方法,將連接實(shí)例直接注入進(jìn)去。這樣example完全不用關(guān)心db連接怎么生成的了。
這就叫依賴(lài)注入,實(shí)現(xiàn)不是在代碼內(nèi)部創(chuàng)建依賴(lài)關(guān)系,而是讓其作為一個(gè)參數(shù)傳遞,這使得我們的程序更容易維護(hù),降低程序代碼的耦合度,實(shí)現(xiàn)一種松耦合。
這還沒(méi)完,我們?cè)偌僭O(shè)example類(lèi)里面除了db還要用到其他外部類(lèi),我們通過(guò):
$example->setDb(Factory::getDb());//注入db連接 $example->setFile(Factory::getFile());//注入文件處理類(lèi) $example->setImage(Factory::getImage());//注入Image處理類(lèi) ...
我們沒(méi)完沒(méi)了的寫(xiě)這么多set?累不累?
ok,為了不用每次寫(xiě)這么多行代碼,我們又去弄了一個(gè)工廠方法:
class Factory { public static function getExample(){ $example = new example(); $example->setDb(Factory::getDb());//注入db連接 $example->setFile(Factory::getFile());//注入文件處理類(lèi) $example->setImage(Factory::getImage());//注入Image處理類(lèi) return $expample; } }
實(shí)例化example時(shí)變?yōu)椋?/p>
$example=Factory::getExample(); $example->getList();
似乎完美了,但是怎么感覺(jué)又回到了上面第一次用工廠方法時(shí)的場(chǎng)景?這確實(shí)不是一個(gè)好的解決方案,所以又提出了一個(gè)概念:容器,又叫做IoC容器、DI容器。
我們本來(lái)是通過(guò)setXXX方法注入各種類(lèi),代碼很長(zhǎng),方法很多,雖然可以通過(guò)一個(gè)工廠方法包裝,但是還不是那么爽,好吧,我們不用setXXX方法了,這樣也就不用工廠方法二次包裝了,那么我們還怎么實(shí)現(xiàn)依賴(lài)注入呢?
這里我們引入一個(gè)約定:在example類(lèi)的構(gòu)造函數(shù)里傳入一個(gè)名為Di $di的參數(shù),如下:
class example { private $_di; function __construct(Di &$di){ $this->_di = $di; } //通過(guò)di容器獲取db實(shí)例 function getList(){ $this->_di->get('db')->query("......");//這里具體sql語(yǔ)句就省略不寫(xiě)了 } } $di = new Di(); $di->set("db",function(){ return new Db("localhost","root","root","test"); }); $example = new example($di); $example->getList();
Di就是IoC容器,所謂容器就是存放我們可能會(huì)用到的各種類(lèi)的實(shí)例,我們通過(guò)$di->set()設(shè)置一個(gè)名為db的實(shí)例,因?yàn)槭峭ㄟ^(guò)回調(diào)函數(shù)的方式傳入的,所以set的時(shí)候并不會(huì)立即實(shí)例化db類(lèi),而是當(dāng)$di->get('db')的時(shí)候才會(huì)實(shí)例化,同樣,在設(shè)計(jì)di類(lèi)的時(shí)候還可以融入單例模式。
這樣我們只要在全局范圍內(nèi)申明一個(gè)Di類(lèi),將所有需要注入的類(lèi)放到容器里,然后將容器作為構(gòu)造函數(shù)的參數(shù)傳入到example,即可在example類(lèi)里面從容器中獲取實(shí)例。當(dāng)然也不一定是構(gòu)造函數(shù),你也可以用一個(gè) setDi(Di $di)的方法來(lái)傳入Di容器,總之約定是你制定的,你自己清楚就行。
這樣一來(lái)依賴(lài)注入以及關(guān)鍵的容器概念已經(jīng)介紹完畢,剩下的就是在實(shí)際中使用并理解它吧!
相關(guān)文章
php抽獎(jiǎng)概率算法(刮刮卡,大轉(zhuǎn)盤(pán))
這篇文章主要為大家詳細(xì)介紹了php中獎(jiǎng)概率算法,可用于刮刮卡,大轉(zhuǎn)盤(pán)等抽獎(jiǎng)算法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-06-06PHP人民幣金額數(shù)字轉(zhuǎn)中文大寫(xiě)的函數(shù)代碼
在網(wǎng)上看到一個(gè)非常有趣的PHP人民幣金額數(shù)字轉(zhuǎn)中文大寫(xiě)的函數(shù),其實(shí)質(zhì)就是數(shù)字轉(zhuǎn)換成中文大寫(xiě),測(cè)試了一下,非常有趣,隨便輸個(gè)數(shù)字,就可以將其大寫(xiě)打印出來(lái),新手朋友們?cè)囈幌掳?/div> 2013-02-02iis6手工創(chuàng)建網(wǎng)站后無(wú)法運(yùn)行php腳本的解決方法
下面小編就為大家?guī)?lái)一篇iis6手工創(chuàng)建網(wǎng)站后無(wú)法運(yùn)行php腳本的解決方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-06-06php+mysql實(shí)現(xiàn)簡(jiǎn)單的增刪改查功能
本文給大家分享的是使用php結(jié)合mysql實(shí)現(xiàn)簡(jiǎn)單的增刪改查的功能的代碼,非常的簡(jiǎn)單實(shí)用,有需要的小伙伴可以參考下。2015-07-07php?overtrue/pinyin拓展實(shí)現(xiàn)漢字轉(zhuǎn)拼音
這篇文章主要為大家介紹了php?overtrue/pinyin拓展實(shí)現(xiàn)漢字轉(zhuǎn)拼音示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11PHP 命名空間和自動(dòng)加載原理與用法實(shí)例分析
這篇文章主要介紹了PHP 命名空間和自動(dòng)加載,結(jié)合實(shí)例形式分析了PHP 命名空間和自動(dòng)加載具體功能、概念、原理與使用技巧,需要的朋友可以參考下2020-04-04最新評(píng)論