Trait是PHP 5.4引入的新概念,看上去既像類又像接口,其實(shí)都不是,Trait可以看做類的部分實(shí)現(xiàn),可以混入一個(gè)或多個(gè)現(xiàn)有的PHP類中,其作用有兩個(gè):表明類可以做什么;提供模塊化實(shí)現(xiàn)。Trait是一種代碼復(fù)用技術(shù),為PHP的單繼承限制提供了一套靈活的代碼復(fù)用機(jī)制。
為什么使用Trait
PHP語言使用一種典型的單繼承模型,在這種模型中,我們先編寫一個(gè)通用的根類,實(shí)現(xiàn)基本的功能,然后擴(kuò)展這個(gè)根類,創(chuàng)建更具體的子類,直接從父類繼承實(shí)現(xiàn)。這叫做繼承層次結(jié)構(gòu),很多編程語言都使用這個(gè)模式。大多數(shù)時(shí)候這種典型的繼承模型能夠良好運(yùn)作,但是如果想讓兩個(gè)無關(guān)的PHP類具有類似的行為,應(yīng)該怎么做呢?
Trait就是為了解決這種問題而誕生的。Trait能夠把模塊化的實(shí)現(xiàn)方式注入多個(gè)無關(guān)的類中,從而提高代碼復(fù)用,符合DRY(Don’t Repeat Yourself)原則。比如Laravel底層用戶認(rèn)證相關(guān)邏輯以及軟刪除實(shí)現(xiàn)等地方都使用了Trait來實(shí)現(xiàn)。以Laravel自帶的AuthController為例,其中的登錄、注冊以及登錄失敗嘗試次數(shù)都是通過Trait實(shí)現(xiàn):
如何創(chuàng)建Trait
創(chuàng)建Trait很簡單,跟創(chuàng)建類有點(diǎn)類似,只不過使用的關(guān)鍵字是trait而不是class,以上述ThrottlesLogin為例:
我們通過trait聲明定義的是一個(gè)Trait,然后我們可以在這個(gè)Trait中像類一樣定義要使用的屬性和方法。
此外Trait支持嵌套和組合,即通過一個(gè)或多個(gè)Trait(多個(gè)用,分隔)組合成一個(gè)Trait,比如
使用多個(gè)Trait可能會(huì)引起命名沖突問題,上面的代碼給出了解決方案:使用insteadof關(guān)鍵字,如果AuthenticatesUsers和RegistersUsers中都定義了redirectPath和getGuard方法,那么將從AuthenticatesUsers中獲取對應(yīng)方法而不是RegistersUsers。另外還可以使用as關(guān)鍵字為方法起個(gè)別名,這樣也可以避免命名沖突。
此外,這里可能沒有完整列出,Trait中還支持定義抽象方法和靜態(tài)方法,其中抽象方法必須在使用它的類中實(shí)現(xiàn)。
這里還需要聲明的一點(diǎn)是調(diào)用方法的優(yōu)先級:調(diào)用類>Trait>父類(如果有的話),方法可以覆蓋,但屬性不行,如果Trait中定義了一個(gè)屬性,如果調(diào)用類中也定義這個(gè)屬性則會(huì)報(bào)錯(cuò)。
如何使用Trait
Trait的使用方法也很簡單,上面已經(jīng)顯示的很清楚明了,即使用use關(guān)鍵字。
可能你已經(jīng)注意到,命名空間和Trait使用的都是use關(guān)鍵字,不同之處在于導(dǎo)入位置,命名空間在類的定義體外導(dǎo)入,而Trait在類的定義體內(nèi)導(dǎo)入。