php 5.4 全新的代碼復(fù)用Trait詳解
從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)單的額例子來進(jìn)行對(duì)比說明。
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ī)則,先來說一下作用域。這個(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)部是都可見的??梢岳斫鉃閡se關(guān)鍵字將Trait的實(shí)現(xiàn)代碼Copy了一份到引用該Trait的類中。
3. Trait中屬性的優(yōu)先級(jí)
說到優(yōu)先級(jí),就必須要有一個(gè)對(duì)比的參照物,這里的參照對(duì)象時(shí)引用Trait的類及其父類。
通過以下的代碼來證明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í)如下:
來自當(dāng)前類的成員覆蓋了 trait 的方法
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)鍵字也用來引用命名空間。兩者的區(qū)別在于,引用Trait時(shí)是在class內(nèi)部使用的。
- PHP 實(shí)現(xiàn)代碼復(fù)用的一個(gè)方法 traits新特性
- PHP中的Trait 特性及作用
- PHP中trait使用方法詳細(xì)介紹
- 淺談PHP中的Trait使用方法
- PHP中Trait及其應(yīng)用詳解
- 簡(jiǎn)單談?wù)凱HP中的trait
- PHP中的traits實(shí)現(xiàn)代碼復(fù)用使用實(shí)例
- PHP Trait代碼復(fù)用類與多繼承實(shí)現(xiàn)方法詳解
- 詳解PHP神奇又有用的Trait
- PHP的Trait機(jī)制原理與用法分析
- PHP中用Trait封裝單例模式的實(shí)現(xiàn)
- PHP Trait功能與用法實(shí)例分析
相關(guān)文章
asp和php下textarea提交大量數(shù)據(jù)發(fā)生丟失的解決方法
2008-01-01PHP細(xì)數(shù)實(shí)現(xiàn)提高并發(fā)能力的方法
這篇文章主要介紹了PHP提高并發(fā)能力有哪些方案,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08php設(shè)計(jì)模式 Composite (組合模式)
將對(duì)象組合成樹形結(jié)構(gòu)以表示"部分-整體"的層次結(jié)構(gòu),使得客戶對(duì)單個(gè)對(duì)象和復(fù)合對(duì)象的使用具有一致性2011-06-06基于PHP安裝zip拓展,以及l(fā)ibzip安裝的問題
下面小編就為大家分享一篇基于PHP安裝zip拓展,以及l(fā)ibzip安裝的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-03-03PHP的curl實(shí)現(xiàn)get,post和cookie(實(shí)例介紹)
本篇文章是對(duì)PHP的curl實(shí)現(xiàn)get,post和cookie的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-06-06解析web文件操作常見安全漏洞(目錄、文件名檢測(cè)漏洞)
本篇文章是對(duì)web文件操作常見安全漏洞(目錄、文件名檢測(cè)漏洞)進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-06-06