在閱讀yii2源碼的時候接觸到了trait,就學習了一下,寫下博客記錄一下。
成都創(chuàng)新互聯(lián)公司堅持“要么做到,要么別承諾”的工作理念,服務領域包括:網(wǎng)站設計制作、成都網(wǎng)站設計、企業(yè)官網(wǎng)、英文網(wǎng)站、手機端網(wǎng)站、網(wǎng)站推廣等服務,滿足客戶于互聯(lián)網(wǎng)時代的東寧網(wǎng)站設計、移動媒體設計的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡建設合作伙伴!自 PHP 5.4.0 起,PHP 實現(xiàn)了代碼復用的一個方法,稱為 traits。
Traits 是一種為類似 PHP 的單繼承語言而準備的代碼復用機制。Trait 為了減少單繼承語言的限制,使開發(fā)人員能夠自由地在不同層次結構內(nèi)獨立的類中復用方法集。Traits 和類組合的語義是定義了一種方式來減少復雜性,避免傳統(tǒng)多繼承和混入類(Mixin)相關的典型問題。
Trait 和一個類相似,但僅僅旨在用細粒度和一致的方式來組合功能。Trait 不能通過它自身來實例化。它為傳統(tǒng)繼承增加了水平特性的組合;也就是說,應用類的成員不需要繼承。
Trait 示例
代碼如下:
優(yōu)先級
從基類繼承的成員被 trait 插入的成員所覆蓋。優(yōu)先順序是來自當前類的成員覆蓋了 trait 的方法,而 trait 則覆蓋了被繼承的方法。
優(yōu)先順序示例
代碼如下:
sayHello(); ?>
以上例程會輸出:Hello World!
從基類繼承的成員被插入的 SayWorld Trait 中的 sayHello 方法所覆蓋。其行為 MyHelloWorld 類中定義的方法一致。優(yōu)先順序是當前類中的方法會覆蓋 trait 方法,而 trait 方法又覆蓋了基類中的方法。
另一個優(yōu)先級順序的例子
代碼如下:
sayHello(); ?>
以上例程會輸出:Hello Universe!
多個 trait
通過逗號分隔,在 use 聲明列出多個 trait,可以都插入到一個類中。
多個 trait 的用法的例子
代碼如下:
sayHello(); $o->sayWorld(); $o->sayExclamationMark(); ?>
以上例程會輸出:Hello World!
沖突的解決
如果兩個 trait 都插入了一個同名的方法,如果沒有明確解決沖突將會產(chǎn)生一個致命錯誤。
為了解決多個 trait 在同一個類中的命名沖突,需要使用 insteadof 操作符來明確指定使用沖突方法中的哪一個。
以上方式僅允許排除掉其它方法,as 操作符可以將其中一個沖突的方法以另一個名稱來引入。
沖突解決的例子
代碼如下:
在本例中 Talker 使用了 trait A 和 B。由于 A 和 B 有沖突的方法,其定義了使用 trait B 中的 smallTalk 以及 trait A 中的 bigTalk。
Aliased_Talker
使用了 as 操作符來定義了 talk 來作為 B 的 bigTalk 的別名。
修改方法的訪問控制
使用 as 語法還可以用來調(diào)整方法的訪問控制。
修改方法的訪問控制的例子
代碼如下:
從 trait 來組成 trait
正如類能夠使用 trait 一樣,其它 trait 也能夠使用 trait。在 trait 定義時通過使用一個或多個 trait,它能夠組合其它 trait 中的部分或全部成員。
從 trait 來組成 trait的例子
代碼如下:
sayHello(); $o->sayWorld(); ?>
以上例程會輸出:Hello World!
Trait 的抽象成員
為了對使用的類施加強制要求,trait 支持抽象方法的使用。
表示通過抽象方法來進行強制要求的例子
代碼如下:
getWorld(); } abstract public function getWorld(); } class MyHelloWorld { private $world; use Hello; public function getWorld() { return $this->world; } public function setWorld($val) { $this->world = $val; } } ?>
Trait 的靜態(tài)成員
Traits 可以被靜態(tài)成員靜態(tài)方法定義。
靜態(tài)變量的例子
代碼如下:
inc(); // echo 1 $p = new C2(); $p->inc(); // echo 1 ?>
靜態(tài)方法的例子
代碼如下:
靜態(tài)變量和靜態(tài)方法的例子
代碼如下:
屬性
Trait 同樣可以定義屬性。
定義屬性的例子
代碼如下:
x; ?>
如果 trait 定義了一個屬性,那類將不能定義同樣名稱的屬性,否則會產(chǎn)生一個錯誤。如果該屬性在類中的定義與在 trait 中的定義兼容(同樣的可見性和初始值)則錯誤的級別是 E_STRICT,否則是一個致命錯誤。
沖突的例子
代碼如下:
Use的不同
不同use的例子
代碼如下:
第一個use是用于 namespace 的 use Foo\\Test,找到的是 \\Foo\\Test,第二個 use 是使用一個trait,找到的是\\Foo\\Bar\\Foo\\Test
。
__CLASS__
和__TRAIT__
__CLASS__
返回 use trait 的 class name,__TRAIT__返回 trait name
示例如下
代碼如下:
testMethod(); //Class: BaseClass //Trait: TestTrait
Trait單例
實例如下
代碼如下:
name = 'foo'; } } class bar { use singleton; private function __construct() { $this->name = 'bar'; } } $foo = foo::getInstance(); echo $foo->name; $bar = bar::getInstance(); echo $bar->name;
調(diào)用trait方法
雖然不很明顯,但是如果Trait的方法可以被定義為在普通類的靜態(tài)方法,就可以被調(diào)用
實例如下
代碼如下: