PHP中Trait及其應用詳解

來源:互聯網
上載者:User
這篇文章主要為大家詳細介紹了PHP中Trait及其應用,具有一定的參考價值,感興趣的小夥伴們可以參考一下

從PHP的5.4.0版本開始,PHP提供了一種全新的代碼複用的概念,那就是Trait。Trait其字面意思是”特性”、”特點”,我們可以理解為,使用Trait關鍵字,可以為PHP中的類添加新的特性。

熟悉物件導向的都知道,軟體開發中常用的代碼複用有繼承和多態兩種方式。在PHP中,只能實現單繼承。而Trait則避免了這點。下面通過簡單的額例子來進行對比說明。

1. 繼承 VS 多態 VS Trait

現在有Publish.php和Answer.php這兩個類。要在其中添加LOG功能,記錄類內部的動作。有以下幾種方案:

繼承
多態
Trait

1.1. 繼承

:

代碼結構如下:

// Log.php<?phpClass Log{ public function startLog() {  // echo ... } public function endLog() {  // echo ... }}

// Publish.php<?phpClass Publish extends Log{} // Answer.php<?phpClass Answer extends Log{}

可以看到繼承的確滿足了要求。但這卻違背了物件導向的原則。而發布(Publish)和回答(Answer)這樣的操作和日誌(Log)之間的關係並不是子類與父類的關係。所以不推薦這樣使用。

1.2. 多態

:

實現代碼:

 // Log.php<?phpInterface Log{ public function startLog(); public function endLog();}

// Publish.php<?phpClass Publish implements Log{ public function startLog() {  // TODO: Implement startLog() method. } public function endLog() {  // TODO: Implement endLog() method. }}

// Answer.php<?phpClass Answer implements Log{ public function startLog() {  // TODO: Implement startLog() method. } public function endLog() {  // TODO: Implement endLog() method. }}

記錄日誌的操作應該都是一樣的,因此,發布(Publish)和回答(Answer)動作中的日誌記錄實現也是一樣的。很明顯,這違背了DRY(Don't Repeat Yourself)原則。所以是不推薦這樣實現的。

1.3. Trait

:

實現代碼如下:

 // Log.php<?phptrait Log{ public function startLog() {  // echo .. } public function endLog() {  // echo .. }}

// Publish.php<?phpclass Publish { use Log;}$publish = new Publish();$publish->startLog();$publish->endLog();

// Answer.php<?phpclass Answer { use Log;}$answer = new Answer();$answer->startLog();$answer->endLog();

可以看到,我們在沒有增加代碼複雜的情況下,實現了代碼的複用。

1.4. 結論

繼承的方式雖然也能解決問題,但其思路違背了物件導向的原則,顯得很粗暴;多態方式也可行,但不符合軟體開發中的DRY原則,增加了維護成本。而Trait方式則避免了上述的不足之處,相對優雅的實現了代碼的複用。

2. Trait的範圍

瞭解了Trait的好處,我們還需要瞭解其實現中的規則,先來說一下範圍。這個比較好證明,實現代碼如下:

 <?phpclass Publish { use Log; public function doPublish() {  $this->publicF();  $this->protectF();  $this->privateF(); }}$publish = new Publish();$publish->doPublish();

執行上述代碼輸出結果如下:
public function
protected function
private function

可以發現,Trait的範圍在引用該Trait類的內部是都可見的。可以理解為use關鍵字將Trait的實現代碼Copy了一份到引用該Trait的類中。

3. Trait中屬性的優先順序

說到優先順序,就必須要有一個對比的參照物,這裡的參照對象時引用Trait的類及其父類。

通過以下的代碼來證明Trait應用中的屬性的優先順序:

 <?phptrait Log{ public function publicF() {  echo __METHOD__ . ' public function' . PHP_EOL; } protected function protectF() {  echo __METHOD__ . ' protected function' . PHP_EOL; }}class Question{ public function publicF() {  echo __METHOD__ . ' public function' . PHP_EOL; } protected function protectF() {  echo __METHOD__ . ' protected function' . PHP_EOL; }}class Publish extends Question{ use Log; public function publicF() {  echo __METHOD__ . ' public function' . PHP_EOL; } public function doPublish() {  $this->publicF();  $this->protectF(); }}$publish = new Publish();$publish->doPublish();

上述代碼的輸出結果如下:
Publish::publicF public function
Log::protectF protected function

通過上面的例子,可以總結出Trait應用中的優先順序如下:
1.來自當前類的成員覆蓋了 trait 的方法
2.trait 覆蓋了被繼承的方法

類成員優先順序為:當前類>Trait>父類

4. Insteadof和As關鍵字

在一個類中,可以引用多個Trait,如下:

 <?phptrait Log{  public function startLog()  {    echo __METHOD__ . ' public function' . PHP_EOL;  }  protected function endLog()  {    echo __METHOD__ . ' protected function' . PHP_EOL;  }}trait Check{  public function parameterCheck($parameters) {    // do sth  }}class Publish extends Question{  use Log,Check;  public function doPublish($para) {    $this->startLog();    $this->parameterCheck($para);    $this->endLog();  }}

通過上面的方式,我們可以在一個類中引用多個Trait。引用多個Trait的時候,就容易出問題了,最常見的問題就是兩個Trait中如果出現了同名的屬性或者方法該怎麼辦呢?這個時候就需要用到Insteadof 和 as 這兩個關鍵字了.請看如下實現代碼:

 <?phptrait Log{  public function parameterCheck($parameters)  {    echo __METHOD__ . ' parameter check' . $parameters . PHP_EOL;  }  public function startLog()  {    echo __METHOD__ . ' public function' . PHP_EOL;  }}trait Check{  public function parameterCheck($parameters)  {    echo __METHOD__ . ' parameter check' . $parameters . PHP_EOL;  }  public function startLog()  {    echo __METHOD__ . ' public function' . PHP_EOL;  }}class Publish{  use Check, Log {    Check::parameterCheck insteadof Log;    Log::startLog insteadof Check;    Check::startLog as csl;  }  public function doPublish()  {    $this->startLog();    $this->parameterCheck('params');    $this->csl();  }}$publish = new Publish();$publish->doPublish();

執行上述代碼,輸出結果如下:
Log::startLog public function
Check::parameterCheck parameter checkparams
Check::startLog public function

就如字面意思一般,insteadof關鍵字用前者取代了後者,as 關鍵字給被取代的方法起了一個別名。

在引用Trait時,使用了use關鍵字,use關鍵字也用來引用命名空間。兩者的區別在於,引用Trait時是在class內部使用的。

以上就是本文的全部內容,希望對大家的學習有所協助。


聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.