如何破解PHP單繼承魔咒?下面本篇文章給大家介紹一下PHP中怎么利用Trait解決單繼承問(wèn)題,希望對(duì)大家有幫助。
我們都知道PHP是單繼承語(yǔ)言,但是有些時(shí)候我們兩個(gè)在業(yè)務(wù)上毫不相干的兩個(gè)類可能存在類似的行為,我們?nèi)绾蝺?yōu)雅的做到DRY(Don’t Repeat Yourself)呢?
答案就是引入Trait
Trait簡(jiǎn)介
Trait是PHP 5.4引入的新概念,看上去既像類又像接口。【推薦學(xué)習(xí):《PHP視頻教程》】
其實(shí)都不是,Trait可以看做類的部分實(shí)現(xiàn),可以混入一個(gè)或多個(gè)現(xiàn)有的PHP類中。
Trait是一種代碼復(fù)用技術(shù),為PHP的單繼承限制提供了一套靈活的代碼復(fù)用機(jī)制。
Trait作用
-
表明類可以做什么;
-
提供模塊化實(shí)現(xiàn)。
單繼承結(jié)構(gòu)
開(kāi)篇我就提到了PHP是單繼承語(yǔ)言
,我們通常的習(xí)慣是:
-
先編寫(xiě)一個(gè)通用的基類,實(shí)現(xiàn)基本的功能,進(jìn)行通用邏輯的封裝,然后擴(kuò)展這個(gè)基類;
-
然后再創(chuàng)建更具體的子類,直接從父類繼承實(shí)現(xiàn)。
這叫方式就是單繼承層次結(jié)構(gòu),很多編程語(yǔ)言都使用這個(gè)模式。
大多數(shù)時(shí)候這種典型的繼承模型能夠良好運(yùn)作,但是如果想讓兩個(gè)無(wú)關(guān)的PHP類具有類似的行為,應(yīng)該怎么做呢?
示例源碼
我查詢了Laravel的源碼,以自帶的LoginController
為例,其中的登錄認(rèn)證就是通過(guò)Trait實(shí)現(xiàn):
創(chuàng)建Trait
創(chuàng)建Trait很簡(jiǎn)單,跟創(chuàng)建類有點(diǎn)類似,只不過(guò)使用的關(guān)鍵字是trait
而不是class
,以上述的AuthenticatesUsers
為例:
我們通過(guò)trait
聲明定義的是一個(gè)Trait,然后我們可以在這個(gè)Trait中像類一樣定義要使用的屬性和方法。
此外Trait支持嵌套和組合,即通過(guò)一個(gè)或多個(gè)Trait(多個(gè)用,分隔)組合成一個(gè)Trait,比如AuthenticatesUsers
就是如此:
注意:Trait中還支持定義抽象方法和靜態(tài)方法,其中抽象方法必須在使用它的類中實(shí)現(xiàn)。
調(diào)用方法的優(yōu)先級(jí):
調(diào)用類>Trait>父類(如果有的話),方法可以覆蓋,但屬性不行。
注意:如果Trait中定義了一個(gè)屬性,如果調(diào)用類中也定義這個(gè)屬性則會(huì)報(bào)錯(cuò)。
使用Trait
Trait的使用方法也很簡(jiǎn)單,上面的截圖示例中已經(jīng)顯示的很清楚明了:使用use
關(guān)鍵字。
引入位置
注意:命名空間和Trait使用的都是use
關(guān)鍵字引入,不同之處在于導(dǎo)入位置,命名空間在類的定義體外導(dǎo)入,而Trait在類的定義體內(nèi)導(dǎo)入。
編譯問(wèn)題
如果Trait和引用Trait的類有相同的屬性和方法,我們?cè)偻獠空{(diào)用時(shí)需要使用instansof
關(guān)鍵字指明方法是類的還是Trait的。
因?yàn)镻HP解釋器在編譯時(shí)會(huì)把Trait復(fù)制到類的定義體中,但是不會(huì)處理這個(gè)操作引入的不兼容問(wèn)題,需要我們來(lái)解決兼容問(wèn)題。