自 PHP 5.4.0 起,PHP 實(shí)現(xiàn)了一種代碼復(fù)用的方法,稱為 trait。
- Trait 是為類似 PHP 的單繼承語(yǔ)言而準(zhǔn)備的一種代碼復(fù)用機(jī)制。Trait 為了減少單繼承語(yǔ)言的限制,使開發(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()?
因此,$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
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://m.rwnh.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)