Trait是PHP 5.4引入的新概念,看上去既像類又像接口,其實都不是,Trait可以看做類的部分實現,可以混入一個或多個現有的PHP類中,其作用有兩個:表明類可以做什么;提供模塊化實現。Trait是一種代碼復用技術,為PHP的單繼承限制提供了一套靈活的代碼復用機制。
為什么使用Trait
PHP語言使用一種典型的單繼承模型,在這種模型中,我們先編寫一個通用的根類,實現基本的功能,然后擴展這個根類,創建更具體的子類,直接從父類繼承實現。這叫做繼承層次結構,很多編程語言都使用這個模式。大多數時候這種典型的繼承模型能夠良好運作,但是如果想讓兩個無關的PHP類具有類似的行為,應該怎么做呢?
Trait就是為了解決這種問題而誕生的。Trait能夠把模塊化的實現方式注入多個無關的類中,從而提高代碼復用,符合DRY(Don’t Repeat Yourself)原則。比如Laravel底層用戶認證相關邏輯以及軟刪除實現等地方都使用了Trait來實現。以Laravel自帶的AuthController為例,其中的登錄、注冊以及登錄失敗嘗試次數都是通過Trait實現:
如何創建Trait
創建Trait很簡單,跟創建類有點類似,只不過使用的關鍵字是trait而不是class,以上述ThrottlesLogin為例:

我們通過trait聲明定義的是一個Trait,然后我們可以在這個Trait中像類一樣定義要使用的屬性和方法。
此外Trait支持嵌套和組合,即通過一個或多個Trait(多個用,分隔)組合成一個Trait,比如

使用多個Trait可能會引起命名沖突問題,上面的代碼給出了解決方案:使用insteadof關鍵字,如果AuthenticatesUsers和RegistersUsers中都定義了redirectPath和getGuard方法,那么將從AuthenticatesUsers中獲取對應方法而不是RegistersUsers。另外還可以使用as關鍵字為方法起個別名,這樣也可以避免命名沖突。
此外,這里可能沒有完整列出,Trait中還支持定義抽象方法和靜態方法,其中抽象方法必須在使用它的類中實現。
這里還需要聲明的一點是調用方法的優先級:調用類>Trait>父類(如果有的話),方法可以覆蓋,但屬性不行,如果Trait中定義了一個屬性,如果調用類中也定義這個屬性則會報錯。
如何使用Trait
Trait的使用方法也很簡單,上面已經顯示的很清楚明了,即使用use關鍵字。
可能你已經注意到,命名空間和Trait使用的都是use關鍵字,不同之處在于導入位置,命名空間在類的定義體外導入,而Trait在類的定義體內導入。




