1. 程式人生 > >【PHP進階學習】——Trait程式碼複用類

【PHP進階學習】——Trait程式碼複用類

前言

眾所周知,一直以來PHP和很多語言一樣是單繼承的語言,但是常常在編碼過程中,我們需要在當前類中使用兩個或兩個以上的其他類的方法,這種情況下繼承就不能實現,而往往採用new方式例項化很多要用到的類,這樣就會很影響程式碼的結構和開發規範。於是Trait類誕生了,它是一種程式碼複用的語法,能夠實現一個類中引用多個其他類的方法。

一、概念

PHP官方手冊對Trait的描述是:
Trait是為類似PHP的單繼承語言而準備的一種程式碼複用機制。Trait為了減少單繼承語言的限制,使開發人員能夠自由地在不同層次結構內獨立的類中複用method。Trait和Class組合的語義定義了一種減少複雜性的方式,避免傳統多繼承和Mixin類相關典型問題。
Trait和Class相似,但僅僅旨在用細粒度和一致的方式來組合功能。無法通過trait自身來例項化。它為傳統繼承增加了水平特性的組合;也就是說,應用的幾個Class之間不需要繼承。

二、Trait類的使用

簡單地講,Trait就是一種不同於繼承的語法,定義一個trait類,在其他類中使用它則是採用use關鍵字,有點類似於名稱空間的用法,但是含義不同。use關鍵字在一個類中引入Trait類後,相當於require或include了一段程式碼進來,不同之處在於use的Trait類與當前類是可以看做同一個類的,即當前類可以用$this關鍵字呼叫Trait類的方法

以下是原理解釋:

可以看出當前類可以簡單地use兩個Trait類,並呼叫其中的方法,而不僅限於繼承,只能使用一個父類的方法。

三、Trait類的訪問控制

我們知道,繼承的方式,如果基類是private修飾控制的,則子類是無法呼叫的。但是Trait
不一樣,因為它類似於Require到當前類中了,所以不管是public、protected或private都是可以直接使用的。示例如下:


四、Trait類的優先順序控制

Trait類與當前使用類、繼承的基類之間的呼叫優先順序順序如下:
當前使用類>Trait類>繼承的基類

當存在同名方法時,會根據優先順序覆蓋掉同名的類。具體示例如下:

1、Trait類覆蓋基類


2、當前類覆蓋Trait類


五、多個Trait類的衝突控制

在PHP中,如果當前類use了兩個Trait類,同時兩個trait類都存在一個同名的方法,此時如果沒有明確解決衝突將會產生一個致命錯誤
對於這種情況,PHP官方給出了兩個解決方案:
1、insteadof關鍵字
:通過該關鍵字指定方法名衝突時該使用哪個Trait類的方法,即:
如果C類use了A、B兩個Trait類,且A、B兩個類都存在a、b方法,則在C類use A、B類時使用insteadof宣告衝突的解決方法即可:
use A, B {
    B::a insteadof A;//a方法衝突時使用B類的a方法而不使用A類的a方法
    A::b insteadof B;//b方法衝突時使用A類的b方法而不使用B類的b方法
}

2、as關鍵字:通過as關鍵字將同名方法指定為一個別名,且僅作用於當前類中。示例如下:
use A, B {
    B::a as c;//宣告B類的a方法為c,作用於該類
    A::b as d;//宣告A類的b方法為d,作用於該類
}

六、與繼承、直接例項化的區別

對於當前一個類需要用到另一個或多個類的方法的情況,我們一般會想到的方式有繼承、直接例項化另外一個或多個類等等的方法,下面來對比一下這些方法和Trait類的區別:
1、繼承方式:對於繼承,可以完美地複用另一個類的一些方法,但是對於需要複用多個類的方法時,PHP是不支援多繼承的,而且只能訪問public和protected方法;
2、與直接例項化的區別:我們也可以在當前類中直接例項化要用到的A類與B類,但是這種方法在控制訪問範圍反面,只允許訪問A、B類中public的方法;

3、使用Trait類則完全將A、B兩個類的方法匯入到當前類中,可以視為當前類的一部分,唯一區別是可以存在於當前類同名的方法,此時由優先順序順序來控制。