99热99这里只有精品6国产,亚洲中文字幕在线天天更新,在线观看亚洲精品国产福利片 ,久久久久综合网

歡迎加入QQ討論群258996829
麥子學(xué)院 頭像
蘋(píng)果6袋
6
麥子學(xué)院

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

發(fā)布時(shí)間:2017-01-05 16:43  回復(fù):0  查看:2135   最后回復(fù):2017-01-05 16:43  

PHP5.4.0版本開(kāi)始,PHP提供了一種全新的代碼復(fù)用的概念,那就是Trait。Trait其字面意思是"特性"、"特點(diǎn)",我們可以理解為,使用Trait關(guān)鍵字,可以為PHP中的類添加新的特性。一起來(lái)看看吧,希望對(duì)大家學(xué)習(xí)php有所幫助。

  熟悉面向?qū)ο蟮亩贾?/span>,軟件開(kāi)發(fā)中常用的代碼復(fù)用有繼承和多態(tài)兩種方式。在PHP,只能實(shí)現(xiàn)單繼承。而Trait則避免了這點(diǎn)。下面通過(guò)簡(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)作。有以下幾種方案:

  1.繼承

  2.多態(tài)

  3.Trait

  1.1. 繼承

  如圖:

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

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

  // Log.php<?phpClass Log{

  public function startLog()

  {

  // echo ...

  }

  public function endLog()

  {

  // echo ...

  }

  }

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

  }

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

  }

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

  1.2. 多態(tài)

  如圖:

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


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

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

  }

  }

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

  1.3. Trait

  如圖:

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


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

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

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

  1.4. 結(jié)論

  繼承的方式雖然也能解決問(wèn)題,但其思路違背了面向?qū)ο蟮脑瓌t,顯得很粗暴;多態(tài)方式也可行,但不符合軟件開(kā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)代碼如下:

  <?phpclass 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)部是都可見(jiàn)的。可以理解為use關(guān)鍵字將Trait的實(shí)現(xiàn)代碼Copy了一份到引用該Trait的類中。

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

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

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

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

  上述代碼的輸出結(jié)果如下:

  Publish::publicF public functionLog::protectF protected function

  通過(guò)上面的例子,可以總結(jié)出Trait應(yīng)用中的優(yōu)先級(jí)如下:

  1.來(lái)自當(dāng)前類的成員覆蓋了 trait 的方法

  2.trait 覆蓋了被繼承的方法

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

  4. InsteadofAs關(guān)鍵字

  在一個(gè)類中,可以引用多個(gè)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();

  }

  }

  通過(guò)上面的方式,我們可以在一個(gè)類中引用多個(gè)Trait。引用多個(gè)Trait的時(shí)候,就容易出問(wèn)題了,最常見(jiàn)的問(wèn)題就是兩個(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 functionCheck::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)部使用的。

 

來(lái)源:SegmentFault

 

您還未登錄,請(qǐng)先登錄

熱門(mén)帖子

最新帖子

?