溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

PHP之Trait特性

發(fā)布時間:2020-07-15 00:02:31 來源:網絡 閱讀:661 作者:hgditren 欄目:web開發(fā)
自 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()?

Trait優(yōu)先級
當前類的函數會覆蓋 trait 的同名函數,trait 會覆蓋父類的同名函數( use trait 相當于當前類直接覆寫了父類的同名函數)

因此,$foo->test()調用的是Trait類中的方法



使用多個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);
        }
                ...
    }
向AI問一下細節(jié)

免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI