PHP中的Trait
PHP提供了一種全新的代碼復(fù)用的概念,那就是Trait。下面一起來(lái)看看!
1. 繼承 VS 多態(tài) VS Trait
現(xiàn)在有Publish.php和Answer.php這兩個(gè)類(lèi)。要在其中添加LOG功能,記錄類(lèi)內(nèi)部的動(dòng)作。有以下幾種方案:
繼承
多態(tài)
Trait
1.1. 繼承
如圖:
代碼結(jié)構(gòu)如下:
// Log.php
Class Log {
public function startLog() {
// echo ...
}
public function endLog() {
// echo ...
}
}
// Publish.php
Class Publish extends Log {
}
// Answer.php
Class Answer extends Log {
}
可以看到繼承的確滿足了要求。但這卻違背了面向?qū)ο蟮脑瓌t。而發(fā)布(Publish)和回答(Answer)這樣的操作和日志(Log)之間的關(guān)系并不是子類(lèi)與父類(lèi)的關(guān)系。所以不推薦這樣使用。
1.2. 多態(tài)
如圖:
實(shí)現(xiàn)代碼:
// Log.php
Interface Log {
public function startLog();
public function endLog();
}
// Publish.php
Class Publish implements Log {
public function startLog() {
// TODO: Implement startLog() method.
}
public function endLog() {
// TODO: Implement endLog() method.
}
}
// Answer.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
trait Log{
public function startLog() {
// echo ..
}
public function endLog() {
// echo ..
}
}
// Publish.php
class Publish {
use Log;
}
$publish = new Publish();
$publish->startLog();
$publish->endLog();
// Answer.php
class 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)代碼如下:
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類(lèi)的內(nèi)部是都可見(jiàn)的。可以理解為use關(guān)鍵字將Trait的實(shí)現(xiàn)代碼Copy了一份到引用該Trait的類(lèi)中。
3. Trait中屬性的優(yōu)先級(jí)
說(shuō)到優(yōu)先級(jí),就必須要有一個(gè)對(duì)比的參照物,這里的參照對(duì)象時(shí)引用Trait的類(lèi)及其父類(lèi)。
通過(guò)以下的代碼來(lái)證明Trait應(yīng)用中的屬性的優(yōu)先級(jí):
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
通過(guò)上面的例子,可以總結(jié)出Trait應(yīng)用中的優(yōu)先級(jí)如下:
來(lái)自當(dāng)前類(lèi)的成員覆蓋了 trait 的方法
trait 覆蓋了被繼承的方法
類(lèi)成員優(yōu)先級(jí)為:當(dāng)前類(lèi)>Trait>父類(lèi)
4. Insteadof和As關(guān)鍵字
在一個(gè)類(lèi)中,可以引用多個(gè)Trait,如下:
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();
}
}
通過(guò)上面的方式,我們可以在一個(gè)類(lèi)中引用多個(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)代碼:
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)部使用的。
【PHP中的Trait】相關(guān)文章:
PHP中的trait是什么08-13
PHP中trait的使用方法08-20
PHP中trait使用方法介紹09-15
PHP中trait的使用方法介紹08-07
PHP中php://input和$-POST的區(qū)別08-26
PHP中的表單處理09-19
Session在PHP中的使用07-24
PHP中l(wèi)ist的方法07-05
PHP中Json應(yīng)用09-05