本文實例講述了PHP面向對象五大原則之依賴倒置原則(DIP)。分享給大家供大家參考,具體如下:
什么是依賴倒置呢?簡單地講就是將依賴關系倒置為依賴接口,具體概念如下:
1.上層模塊不應該依賴于下層模塊,它們共同依賴于一個抽象(父類不能依賴子類,它們都要依賴于抽象類)
2.抽象不能依賴于具體,具體應該要依賴于抽象。
注意,這里的接口不是狹義的接口。
為什么要依賴接口?因為接口體現對問題的抽象,同時由于抽象一般是相對穩定的或者是相對變化不頻繁的,而具體是易變的。因此依賴抽象是實現代碼擴展和運行期內綁定(多態)的基礎:只要實現了該抽象類的子類,都可以被類的使用都使用。這里,強調一下擴展性這個概念。通常擴展性指對已知行為的擴展,在講述接口時,也提到過,接口應該是相對的。這就告訴我們,無論使用多么先進的設計模式,也無法做到不需要修改代碼即可達到不變應萬變的地上。在面向對象的這五大原則里,我認為依賴倒置是最難理解,也是最難實現的。
這里以雇員類為例
?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實現;在workB中,work轉而依賴抽象,這樣可以把需要的對象通過參數傳入。上述代碼通過接口,實現了一定程度的解耦,但仍然是有限的。不僅是使用接口,使用工廠等也能實現一定程度的解耦和依賴倒置。
在workB中,teacher實例通過set方法傳入,從而實現了工廠模式。由于這樣的實現仍然是硬編碼的,為了實現代碼的進一步擴展,把這個依賴關系寫在配置文件里,指明workB需要一個teacher對象,專門由一個程序配置是否正確(如所依賴的類文件是否存在)以及加載配置中所依賴的實現,這個檢測程序,就稱為IOC容器。
很多文章里看到IOC(Inversion of Control)概念,實際上,IOC是依賴倒置原則(Dependence Inversion Principle,DIP)的同義詞。而在提IOC的時候,你可能還會看到有人提起DI等概念。DI,即依賴注入,一般認為,依賴注入(DI)和依賴查找(DS)是IOC的兩種實現。不過隨著某些概論的演化,這幾個概念之間的關系也變得很模糊,也有人認為IOC就是DI。有人認為,依賴注入的描述比起IOC來更貼切,這里不糾纏于這幾個概念之間的關系。
在經典的J2EE設計里,通常把DAO層和Servicen層細分為接口層和實現層,然后在配置文件里進行所依賴關系的配置,這是最常見的DIP的應用。Spring框架就是一個很好的IOC容器,把控制權從代碼剝離到IOC窗口,這里是通過XML配置文件實現的,Spring在執行期間根據配置文件的設定,建立對象之間的依賴關系。
如下面的代碼所示
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>
但是這樣的設置一樣存在問題,配置文件會變得越來越大,其間關系會越來越復雜。同樣逃脫不了隨著應用和業務的改變,不斷修改代碼的惡魘(這里認為配置文件是代碼的一部分。并且在實際開發中,很少存在單純修改配置文件的情況。一般配置文件修改了,代碼也會做相應的修改)
在PHP里,也有類似模仿Spring的實現,即把依賴關系寫在了配置文件里,通過配置文件來產生需要的對象。我覺得這樣的代碼還是為了實現而實現。在Srping里,配置文件里配置的不僅僅是一個類運行時的依賴關系,還可以實現事務管理、AOP、延遲加載等。而PHP要實現上面的種種特性,其消耗是巨大的。從語言層面講,PHP這種動態腳本語言在實現一些多態特性上和編譯型的語言不同。其次PHP作為敏捷性的開發語言,更強調快速開發、邏輯清晰、代碼更簡單易懂,如果再附加了各種設計模式的框架,從技術實現和運行效率上來看,都是不可取的。依賴倒置的核心原則是解耦。如果脫離這個最原始的原則,那就是本末倒置。
事實上,很多的設計模式里已經隱含了依賴倒置原則我們也在有意無意地做著一些依賴反轉的工作。只是作為PHP,目前還沒有一個比較完善的IOC容器,或許是PHP根本不需要。
如果滿足DIP:
1.每個較高層次類都為它所需要的服務提出一個接口聲明,較低層次類實現實現這個接口。
2.每個高層次類都通過該抽象接口使用服務。
更多關于PHP相關內容感興趣的讀者可查看本站專題:《php面向對象程序設計入門教程》、《PHP數組(Array)操作技巧大全》、《PHP基本語法入門教程》、《PHP運算與運算符用法總結》、《php字符串(string)用法總結》、《php+mysql數據庫操作入門教程》及《php常見數據庫操作技巧匯總》
希望本文所述對大家PHP程序設計有所幫助。
您可能感興趣的文章:- PHP面向對象五大原則之里氏替換原則(LSP)詳解
- 舉例解析Java的設計模式編程中里氏替換原則的意義
- 詳解Java設計模式編程中的里氏替換原則
- 深入理解JavaScript系列(8) S.O.L.I.D五大原則之里氏替換原則LSP
- PHP面向對象五大原則之接口隔離原則(ISP)詳解
- PHP面向對象五大原則之開放-封閉原則(OCP)詳解
- PHP面向對象五大原則之單一職責原則(SRP)詳解
- PHP基于面向對象實現的留言本功能實例
- PHP面向對象之里氏替換原則簡單示例