首頁  >  文章  >  後端開發  >  PHP中Trait及其應用詳解

PHP中Trait及其應用詳解

墨辰丷
墨辰丷原創
2018-05-25 14:37:431821瀏覽

這篇文章主要為大家詳細介紹了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
<?php
Class Log
{
 public function startLog()
 {
  // echo ...
 }

 public function endLog()
 {
  // echo ...
 }
}

// Publish.php
<?php
Class Publish extends Log
{

} // Answer.php
<?php
Class Answer extends Log
{

}

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

1.2. 多型態

 

如圖:


 

實作程式碼:

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


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

##
// Answer.php
<?php
Class 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
<?php
trait Log{
 public function startLog() {
  // echo ..
 }
 public function endLog() {
  // echo ..
 }
}

#

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

##

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

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


1.4. 結論

 


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

2. Trait的作用域
 

#了解了Trait的好處,我們還需要了解其實現中的規則,先來說一下作用域。這個比較好證明,實作程式碼如下:


 <?php
class 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應用程式中的屬性的優先權:


#
 <?php
trait Log
{
 public function publicF()
 {
  echo __METHOD__ . &#39; public function&#39; . PHP_EOL;
 }
 protected function protectF()
 {
  echo __METHOD__ . &#39; protected function&#39; . PHP_EOL;
 }
}

class Question
{
 public function publicF()
 {
  echo __METHOD__ . &#39; public function&#39; . PHP_EOL;
 }
 protected function protectF()
 {
  echo __METHOD__ . &#39; protected function&#39; . PHP_EOL;
 }
}

class Publish extends Question
{
 use Log;

 public function publicF()
 {
  echo __METHOD__ . &#39; public function&#39; . 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,如下:############
 <?php
trait Log
{
  public function startLog()
  {
    echo __METHOD__ . &#39; public function&#39; . PHP_EOL;
  }
  protected function endLog()
  {
    echo __METHOD__ . &#39; protected function&#39; . 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 這兩個關鍵字了.請看如下實作程式碼:######

 <?php

trait Log
{
  public function parameterCheck($parameters)
  {
    echo __METHOD__ . &#39; parameter check&#39; . $parameters . PHP_EOL;
  }

  public function startLog()
  {
    echo __METHOD__ . &#39; public function&#39; . PHP_EOL;
  }
}

trait Check
{
  public function parameterCheck($parameters)
  {
    echo __METHOD__ . &#39; parameter check&#39; . $parameters . PHP_EOL;
  }

  public function startLog()
  {
    echo __METHOD__ . &#39; public function&#39; . 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(&#39;params&#39;);
    $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内部使用的。

以上就是本文的全部内容,希望对大家的学习有所帮助。


相关推荐:

关于PHP中的trait简单介绍

PHP中trait使用方法图文详解

php的Traits属性以及基本用法

以上是PHP中Trait及其應用詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn