PHP之Trait特性

自 PHP 5.4.0 起,PHP 實(shí)現(xiàn)了一種代碼復(fù)用的方法,稱為 trait。
  • Trait 是為類似 PHP 的單繼承語(yǔ)言而準(zhǔn)備的一種代碼復(fù)用機(jī)制。Trait 為了減少單繼承語(yǔ)言的限制,使開(kāi)發(fā)人員能夠自由地在不同層次結(jié)構(gòu)內(nèi)獨(dú)立的類中復(fù)用 method。Trait 和 Class 組合的語(yǔ)義定義了一種減少?gòu)?fù)雜性的方式,避免傳統(tǒng)多繼承和 Mixin 類相關(guān)典型問(wèn)題。
  • Trait 和 Class 相似,但僅僅旨在用細(xì)粒度和一致的方式來(lái)組合功能。 無(wú)法通過(guò) trait 自身來(lái)實(shí)例化。它為傳統(tǒng)繼承增加了水平特性的組合;也就是說(shuō),應(yīng)用的幾個(gè) Class 之間不需要繼承。
<?php

//日志類
class Logger
{
    //讀取日志信息
    public function log(string $message, int $level)
    {
        echo "[message]:{$message}" . PHP_EOL;
        echo "[level]:{$level}" . PHP_EOL;
    }
}

//擴(kuò)展日志功能
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延時(shí)靜態(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來(lái)擴(kuò)展Foo類增加日志功能;
    • Trait Loggable類中含有test()方法;
    • Foo類繼承Base類,其中Base類中含有test();
    • 那么問(wèn)題來(lái)了:$foo->test()到底調(diào)用的是繼承自父類test(),還是Trait類中的test()?

Trait優(yōu)先級(jí)
當(dāng)前類的函數(shù)會(huì)覆蓋 trait 的同名函數(shù),trait 會(huì)覆蓋父類的同名函數(shù)( use trait 相當(dāng)于當(dāng)前類直接覆寫(xiě)了父類的同名函數(shù))

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

成都創(chuàng)新互聯(lián)從2013年成立,是專業(yè)互聯(lián)網(wǎng)技術(shù)服務(wù)公司,擁有項(xiàng)目成都網(wǎng)站設(shè)計(jì)、成都網(wǎng)站建設(shè)網(wǎng)站策劃,項(xiàng)目實(shí)施與項(xiàng)目整合能力。我們以讓每一個(gè)夢(mèng)想脫穎而出為使命,1280元加格達(dá)奇做網(wǎng)站,已為上家服務(wù),為加格達(dá)奇各地企業(yè)和個(gè)人服務(wù),聯(lián)系電話:13518219792



使用多個(gè)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(取代) 操作符來(lái)明確指定使用沖突方法中的哪一個(gè)
  • as 操作符可以 為某個(gè)方法引入別名。 注意,as 操作符不會(huì)對(duì)方法進(jìn)行重命名,也不會(huì)影響其方法。

最終代碼:

class TopBoss
{
    use Alibaba, Tencent {
        Tencent::getCEO insteadof Alibaba;//指定沖突時(shí),使用誰(shuí)
        Tencent::getAddress insteadof Alibaba;
        Alibaba::getAddress as getA;//取別名,可以通過(guò)別名調(diào)用
        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);
        }
                ...
    }

新聞標(biāo)題:PHP之Trait特性
網(wǎng)頁(yè)地址:http://muchs.cn/article6/jejsig.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供動(dòng)態(tài)網(wǎng)站、網(wǎng)站改版網(wǎng)站排名、App設(shè)計(jì)響應(yīng)式網(wǎng)站、微信公眾號(hào)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)

成都網(wǎng)站建設(shè)