這篇文章主要為大家詳細介紹了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. } }
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的作用域
<?php class Publish { use Log; public function doPublish() { $this->publicF(); $this->protectF(); $this->privateF(); } } $publish = new Publish(); $publish->doPublish();
public function
private function
#可以發現,Trait的作用域在引用該Trait類別的內部是都可見的。可以理解為use關鍵字將Trait的實作程式碼Copy了一份到引用該Trait的類別中。
3. Trait中屬性的優先順序
#說到優先級,就必須要有一個對比的參考物,這裡的參考物件時引用Trait的類別及其父類別。
透過以下的程式碼證明Trait應用程式中的屬性的優先權:
<?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();
Publish::publicF public function
Log::protectF protected function
透過上面的範例,可以總結出Trait應用程式中的優先權如下:
1.來自目前類別的成員覆寫了trait 的方法2.trait 覆寫了被繼承的方法
類別成員優先權為:目前類別>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(); } }#########透過上面的方式,我們可以在一個類別中引用多個Trait。引用多個Trait的時候,就容易出問題了,最常見的問題就是兩個Trait中如果出現了同名的屬性或者方法該怎麼辦呢?這個時候就需要用到Insteadof 和as 這兩個關鍵字了.請看如下實作程式碼:######
<?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();
执行上述代码,输出结果如下:
Log::startLog public function
Check::parameterCheck parameter checkparams
Check::startLog public function
就如字面意思一般,insteadof关键字用前者取代了后者,as 关键字给被取代的方法起了一个别名。
在引用Trait时,使用了use关键字,use关键字也用来引用命名空间。两者的区别在于,引用Trait时是在class内部使用的。
以上就是本文的全部内容,希望对大家的学习有所帮助。
相关推荐:
以上是PHP中Trait及其應用詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!