PHP面向?qū)ο笪宕笤瓌t之依賴倒置原則(DIP)詳解
本文實(shí)例講述了PHP面向?qū)ο笪宕笤瓌t之依賴倒置原則(DIP)。分享給大家供大家參考,具體如下:
什么是依賴倒置呢?簡單地講就是將依賴關(guān)系倒置為依賴接口,具體概念如下:
1.上層模塊不應(yīng)該依賴于下層模塊,它們共同依賴于一個(gè)抽象(父類不能依賴子類,它們都要依賴于抽象類)
2.抽象不能依賴于具體,具體應(yīng)該要依賴于抽象。
注意,這里的接口不是狹義的接口。
為什么要依賴接口?因?yàn)榻涌隗w現(xiàn)對問題的抽象,同時(shí)由于抽象一般是相對穩(wěn)定的或者是相對變化不頻繁的,而具體是易變的。因此依賴抽象是實(shí)現(xiàn)代碼擴(kuò)展和運(yùn)行期內(nèi)綁定(多態(tài))的基礎(chǔ):只要實(shí)現(xiàn)了該抽象類的子類,都可以被類的使用都使用。這里,強(qiáng)調(diào)一下擴(kuò)展性這個(gè)概念。通常擴(kuò)展性指對已知行為的擴(kuò)展,在講述接口時(shí),也提到過,接口應(yīng)該是相對的。這就告訴我們,無論使用多么先進(jìn)的設(shè)計(jì)模式,也無法做到不需要修改代碼即可達(dá)到不變應(yīng)萬變的地上。在面向?qū)ο蟮倪@五大原則里,我認(rèn)為依賴倒置是最難理解,也是最難實(shí)現(xiàn)的。
這里以雇員類為例
<?php interface employee { public function working(); } class teacher implements employee { public function working() { echo 'teaching...'; } } class coder implements employee { public function working() { echo 'coding...'; } } class workA { public function work() { $teacher = new teacher(); $teacher->working(); } } class workB { private $e; public function set(employee $e) { $this->e = $e; } public function work() { $this->e->working(); } } $worka = new workA; $worka->work(); $workb = new workB; $workb->set(new teacher()); $workb->work();
在workA中,work方法依賴于teacher實(shí)現(xiàn);在workB中,work轉(zhuǎn)而依賴抽象,這樣可以把需要的對象通過參數(shù)傳入。上述代碼通過接口,實(shí)現(xiàn)了一定程度的解耦,但仍然是有限的。不僅是使用接口,使用工廠等也能實(shí)現(xiàn)一定程度的解耦和依賴倒置。
在workB中,teacher實(shí)例通過set方法傳入,從而實(shí)現(xiàn)了工廠模式。由于這樣的實(shí)現(xiàn)仍然是硬編碼的,為了實(shí)現(xiàn)代碼的進(jìn)一步擴(kuò)展,把這個(gè)依賴關(guān)系寫在配置文件里,指明workB需要一個(gè)teacher對象,專門由一個(gè)程序配置是否正確(如所依賴的類文件是否存在)以及加載配置中所依賴的實(shí)現(xiàn),這個(gè)檢測程序,就稱為IOC容器。
很多文章里看到IOC(Inversion of Control)概念,實(shí)際上,IOC是依賴倒置原則(Dependence Inversion Principle,DIP)的同義詞。而在提IOC的時(shí)候,你可能還會看到有人提起DI等概念。DI,即依賴注入,一般認(rèn)為,依賴注入(DI)和依賴查找(DS)是IOC的兩種實(shí)現(xiàn)。不過隨著某些概論的演化,這幾個(gè)概念之間的關(guān)系也變得很模糊,也有人認(rèn)為IOC就是DI。有人認(rèn)為,依賴注入的描述比起IOC來更貼切,這里不糾纏于這幾個(gè)概念之間的關(guān)系。
在經(jīng)典的J2EE設(shè)計(jì)里,通常把DAO層和Servicen層細(xì)分為接口層和實(shí)現(xiàn)層,然后在配置文件里進(jìn)行所依賴關(guān)系的配置,這是最常見的DIP的應(yīng)用。Spring框架就是一個(gè)很好的IOC容器,把控制權(quán)從代碼剝離到IOC窗口,這里是通過XML配置文件實(shí)現(xiàn)的,Spring在執(zhí)行期間根據(jù)配置文件的設(shè)定,建立對象之間的依賴關(guān)系。
如下面的代碼所示
<bean scopre="prototype" class="cn.notebook.action.NotebookListOtherAction" id="notebookListOtherAction"> <property ref="userReplyService" name="userReplyService" /> <property ref="userService" name="userService" /> <property ref="permissionService" name="permissionService" /> <property ref="friendService" name="friendService" /> </bean>
但是這樣的設(shè)置一樣存在問題,配置文件會變得越來越大,其間關(guān)系會越來越復(fù)雜。同樣逃脫不了隨著應(yīng)用和業(yè)務(wù)的改變,不斷修改代碼的惡魘(這里認(rèn)為配置文件是代碼的一部分。并且在實(shí)際開發(fā)中,很少存在單純修改配置文件的情況。一般配置文件修改了,代碼也會做相應(yīng)的修改)
在PHP里,也有類似模仿Spring的實(shí)現(xiàn),即把依賴關(guān)系寫在了配置文件里,通過配置文件來產(chǎn)生需要的對象。我覺得這樣的代碼還是為了實(shí)現(xiàn)而實(shí)現(xiàn)。在Srping里,配置文件里配置的不僅僅是一個(gè)類運(yùn)行時(shí)的依賴關(guān)系,還可以實(shí)現(xiàn)事務(wù)管理、AOP、延遲加載等。而PHP要實(shí)現(xiàn)上面的種種特性,其消耗是巨大的。從語言層面講,PHP這種動態(tài)腳本語言在實(shí)現(xiàn)一些多態(tài)特性上和編譯型的語言不同。其次PHP作為敏捷性的開發(fā)語言,更強(qiáng)調(diào)快速開發(fā)、邏輯清晰、代碼更簡單易懂,如果再附加了各種設(shè)計(jì)模式的框架,從技術(shù)實(shí)現(xiàn)和運(yùn)行效率上來看,都是不可取的。依賴倒置的核心原則是解耦。如果脫離這個(gè)最原始的原則,那就是本末倒置。
事實(shí)上,很多的設(shè)計(jì)模式里已經(jīng)隱含了依賴倒置原則我們也在有意無意地做著一些依賴反轉(zhuǎn)的工作。只是作為PHP,目前還沒有一個(gè)比較完善的IOC容器,或許是PHP根本不需要。
如果滿足DIP:
1.每個(gè)較高層次類都為它所需要的服務(wù)提出一個(gè)接口聲明,較低層次類實(shí)現(xiàn)實(shí)現(xiàn)這個(gè)接口。
2.每個(gè)高層次類都通過該抽象接口使用服務(wù)。
更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《php面向?qū)ο蟪绦蛟O(shè)計(jì)入門教程》、《PHP數(shù)組(Array)操作技巧大全》、《PHP基本語法入門教程》、《PHP運(yùn)算與運(yùn)算符用法總結(jié)》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫操作入門教程》及《php常見數(shù)據(jù)庫操作技巧匯總》
希望本文所述對大家PHP程序設(shè)計(jì)有所幫助。
相關(guān)文章
PHP laravel使用自定義郵件類實(shí)現(xiàn)發(fā)送郵件
這篇文章主要為大家詳細(xì)介紹了PHP laravel如何通過自定義郵件類實(shí)現(xiàn)發(fā)送郵件功能,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以嘗試一下2022-10-10學(xué)習(xí)php設(shè)計(jì)模式 php實(shí)現(xiàn)橋梁模式(bridge)
這篇文章主要介紹了php設(shè)計(jì)模式中的橋梁模式,使用php實(shí)現(xiàn)橋梁模式,感興趣的小伙伴們可以參考一下2015-12-12php 智能404跳轉(zhuǎn)代碼,適合換域名沒改變目錄的網(wǎng)站
適合于換域名,但是目錄沒有改變的網(wǎng)站.也可以用做301定向.轉(zhuǎn)自于落伍,收集過來.怕以后需要.需要的兄弟直接copy過去即可.2010-06-06PHP FileSystem 文件系統(tǒng)常用api整理總結(jié)
這篇文章主要介紹了PHP FileSystem 文件系統(tǒng)常用api,結(jié)合實(shí)例形式整理總結(jié)了php文件系統(tǒng)常用函數(shù)的功能、使用技巧與相關(guān)操作注意事項(xiàng),需要的朋友可以參考下2019-07-07php遞歸創(chuàng)建和刪除文件夾的代碼小結(jié)
有時(shí)候需要遞歸創(chuàng)建和刪除文件夾,那么就可以參考下面的代碼2012-04-04