自 PHP 5.4.0 起,PHP 實現了一種代碼複用的方法,稱爲 trait。
- Trait 是爲相似 PHP 的單繼承語言而準備的一種代碼複用機制。Trait 爲了減小單繼承語言的限制,使開發人員可以自由地在不一樣層次結構內獨立的類中複用 method。Trait 和 Class 組合的語義定義了一種減小複雜性的方式,避免傳統多繼承和 Mixin 類相關典型問題。
- Trait 和 Class 類似,但僅僅旨在用細粒度和一致的方式來組合功能。 沒法經過 trait 自身來實例化。它爲傳統繼承增長了水平特性的組合;也就是說,應用的幾個 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延時靜態綁定 } } 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類中的方法php
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();
解決方案ide
- 使用 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中的代碼示例:this
<?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); } ... }