欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

PHP中Trait及其應(yīng)用詳解

 更新時(shí)間:2017年02月14日 15:40:16   作者:lanffy  
這篇文章主要為大家詳細(xì)介紹了PHP中Trait及其應(yīng)用,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下

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

熟悉面向?qū)ο蟮亩贾?軟件開發(fā)中常用的代碼復(fù)用有繼承和多態(tài)兩種方式。在PHP中,只能實(shí)現(xiàn)單繼承。而Trait則避免了這點(diǎn)。下面通過簡(jiǎn)單的額例子來(lái)進(jìn)行對(duì)比說(shuō)明。 

1. 繼承 VS 多態(tài) VS Trait 

現(xiàn)在有Publish.php和Answer.php這兩個(gè)類。要在其中添加LOG功能,記錄類內(nèi)部的動(dòng)作。有以下幾種方案:

 繼承
 多態(tài)
 Trait 

1.1. 繼承 

如圖:

代碼結(jié)構(gòu)如下:

// 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
{

} 

可以看到繼承的確滿足了要求。但這卻違背了面向?qū)ο蟮脑瓌t。而發(fā)布(Publish)和回答(Answer)這樣的操作和日志(Log)之間的關(guān)系并不是子類與父類的關(guān)系。所以不推薦這樣使用。 

1.2. 多態(tài) 

如圖:

 

實(shí)現(xiàn)代碼:

 // 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.
 }
} 

記錄日志的操作應(yīng)該都是一樣的,因此,發(fā)布(Publish)和回答(Answer)動(dòng)作中的日志記錄實(shí)現(xiàn)也是一樣的。很明顯,這違背了DRY(Don't Repeat Yourself)原則。所以是不推薦這樣實(shí)現(xiàn)的。 

1.3. Trait 

如圖:

 

實(shí)現(xiàn)代碼如下:

 // 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(); 

可以看到,我們?cè)跊]有增加代碼復(fù)雜的情況下,實(shí)現(xiàn)了代碼的復(fù)用。 

1.4. 結(jié)論 

繼承的方式雖然也能解決問題,但其思路違背了面向?qū)ο蟮脑瓌t,顯得很粗暴;多態(tài)方式也可行,但不符合軟件開發(fā)中的DRY原則,增加了維護(hù)成本。而Trait方式則避免了上述的不足之處,相對(duì)優(yōu)雅的實(shí)現(xiàn)了代碼的復(fù)用。 

2. Trait的作用域 

了解了Trait的好處,我們還需要了解其實(shí)現(xiàn)中的規(guī)則,先來(lái)說(shuō)一下作用域。這個(gè)比較好證明,實(shí)現(xiàn)代碼如下:

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

執(zhí)行上述代碼輸出結(jié)果如下:
public function
protected function
private function

可以發(fā)現(xiàn),Trait的作用域在引用該Trait類的內(nèi)部是都可見的。可以理解為use關(guān)鍵字將Trait的實(shí)現(xiàn)代碼Copy了一份到引用該Trait的類中。 

3. Trait中屬性的優(yōu)先級(jí)

說(shuō)到優(yōu)先級(jí),就必須要有一個(gè)對(duì)比的參照物,這里的參照對(duì)象時(shí)引用Trait的類及其父類。 

通過以下的代碼來(lái)證明Trait應(yīng)用中的屬性的優(yōu)先級(jí):

 <?php
trait 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(); 

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

通過上面的例子,可以總結(jié)出Trait應(yīng)用中的優(yōu)先級(jí)如下:
 1.來(lái)自當(dāng)前類的成員覆蓋了 trait 的方法
 2.trait 覆蓋了被繼承的方法 

類成員優(yōu)先級(jí)為:當(dāng)前類>Trait>父類

4. Insteadof和As關(guān)鍵字 

在一個(gè)類中,可以引用多個(gè)Trait,如下:

 <?php
trait 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();
  }
} 

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

 <?php

trait 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(); 

執(zhí)行上述代碼,輸出結(jié)果如下:
 Log::startLog public function
Check::parameterCheck parameter checkparams
Check::startLog public function

就如字面意思一般,insteadof關(guān)鍵字用前者取代了后者,as 關(guān)鍵字給被取代的方法起了一個(gè)別名。 

在引用Trait時(shí),使用了use關(guān)鍵字,use關(guān)鍵字也用來(lái)引用命名空間。兩者的區(qū)別在于,引用Trait時(shí)是在class內(nèi)部使用的。

 以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

最新評(píng)論