您好,登錄后才能下訂單哦!
自 PHP 5.4.0 起,PHP 實現(xiàn)了一種代碼復用的方法,稱為 trait。
- Trait 是為類似 PHP 的單繼承語言而準備的一種代碼復用機制。Trait 為了減少單繼承語言的限制,使開發(fā)人員能夠自由地在不同層次結構內獨立的類中復用 method。Trait 和 Class 組合的語義定義了一種減少復雜性的方式,避免傳統(tǒng)多繼承和 Mixin 類相關典型問題。
- Trait 和 Class 相似,但僅僅旨在用細粒度和一致的方式來組合功能。 無法通過 trait 自身來實例化。它為傳統(tǒng)繼承增加了水平特性的組合;也就是說,應用的幾個 Class 之間不需要繼承。
<?php
//日志類
class Logger
{
//讀取日志信息
public function log(string $message, int $level)
{
echo "[message]:{$message}" . PHP_EOL;
echo "[level]:{$level}" . PHP_EOL;
}
}
//擴展日志功能
trait Loggable
{
protected $logger;
/**
* 記錄日志
* @param DemoLogger $logger
*/
public function setLogger(Logger $logger)
{
$this->logger = $logger;
}
/**
* 讀取日志
* @param string $message
* @param int $level
*/
public function getLog(string $message, int $level)
{
$this->logger->log($message, $level);
}
public function test()
{
echo 'trait test' . PHP_EOL;
}
}
//基類
class Base
{
public static $className = 'Base';
public function test()
{
echo static::getClassName() . ' test' . PHP_EOL;
}
//獲取類名稱
public static function getClassName(): string
{
//return self::$className;
return static::$className;//static延時靜態(tài)綁定
}
}
class Foo extends Base
{
public static $className = 'Foo';
use Loggable;
}
$foo = new Foo;
$foo->setLogger(new Logger);
$foo->getLog('trait works', 1);//打印日志信息
$foo->test(); //trait test
分析這里 $foo->test()
- Foo類中使用use Loggable來擴展Foo類增加日志功能;
- Trait Loggable類中含有test()方法;
- Foo類繼承Base類,其中Base類中含有test();
- 那么問題來了:$foo->test()到底調用的是繼承自父類test(),還是Trait類中的test()?
因此,$foo->test()調用的是Trait類中的方法
Trait Alibaba
{
public function getCEO(): string
{
return '阿里巴巴CEO:馬云' . PHP_EOL;
}
public function getAddress(): string
{
return '阿里巴巴總部位于杭州' . PHP_EOL;
}
}
Trait Tencent
{
public function getCEO(): string
{
return '騰訊CEO:馬化騰' . PHP_EOL;
}
public function getAddress(): string
{
return '騰訊總部位于深圳' . PHP_EOL;
}
}
class TopBoss
{
use Alibaba, Tencent;
}
$MaBoss = new TopBoss();
echo $MaBoss->getCEO();
echo $MaBoss->getAddress();
解決方案
- 使用 insteadof(取代) 操作符來明確指定使用沖突方法中的哪一個
- as 操作符可以 為某個方法引入別名。 注意,as 操作符不會對方法進行重命名,也不會影響其方法。
最終代碼:
class TopBoss
{
use Alibaba, Tencent {
Tencent::getCEO insteadof Alibaba;//指定沖突時,使用誰
Tencent::getAddress insteadof Alibaba;
Alibaba::getAddress as getA;//取別名,可以通過別名調用
Alibaba::getCEO as getC;
}
}
$MaBoss = new TopBoss();
echo $MaBoss->getCEO();//騰訊CEO:馬化騰
echo $MaBoss->getAddress();//騰訊總部位于深圳
echo $superBoss->getC();//阿里巴巴CEO:馬云
echo $superBoss->getA();//阿里巴巴總部位于杭州
Laravel中的代碼示例:
<?php
namespace Illuminate\Support;
use ArrayAccess;
class Optional implements ArrayAccess
{
use Traits\Macroable {
__call as macroCall;
}
...
/**
* Dynamically pass a method to the underlying object.
*
* @param string $method
* @param array $parameters
* @return mixed
*/
public function __call($method, $parameters)
{
if (static::hasMacro($method)) {
return $this->macroCall($method, $parameters);
}
...
}
免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。