PHP之Trait特性

自 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()?

Trait優先級

當前類的函數會覆蓋 trait 的同名函數,trait 會覆蓋父類的同名函數( use trait 至關於當前類直接覆寫了父類的同名函數)

所以,$foo->test()調用的是Trait類中的方法php



使用多個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();

解決方案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);
        }
                ...
    }
相關文章
相關標籤/搜索