相信你們對trait已經不陌生了,早在5.4時,trait就已經出如今了PHP的新特性中。固然,自己trait也是特性的意思,但這個特性的主要能力就是爲了代碼的複用。php
咱們都知道,PHP是現代化的面嚮對象語言。爲了解決C++多重繼承的混亂問題,大部分語言都是單繼承多接口的形式,但這也會讓一些能夠複用的代碼必須經過組合的方式來實現,若是要用到組合,不可避免的就要實例化類或者使用靜態方法,無形中增長了內存的佔用。而PHP爲了解決這個問題,就正式推出了trait能力。你能夠把它看作是組合能力的一種變體。git
trait A { public $a = 'A'; public function testA() { echo 'This is ' . $this->a; } } class classA { use A; } class classB { use A; public function __construct() { $this->a = 'B'; } } $a = new classA(); $b = new classB(); $a->testA(); $b->testA();
從上述代碼中,咱們能夠看出,trait能夠給應用於任意一個類中,並且能夠定義變量,很是方便。trait最須要注意的是關於同名方法的重載優先級問題。github
trait B { function test(){ echo 'This is trait B!'; } } trait C { function test(){ echo 'This is trait C!'; } } class testB{ use B, C; function test(){ echo 'This is class testB!'; } } $b = new testB(); $b->test(); // This is class testB! // class testC{ // use B, C; // } // $c = new testC(); // $c->test(); // Fatal error: Trait method test has not been applied, because there are collisions with other trait methods on testC
在這裏,咱們的類中重載了test()方法,這裏輸出的就是類中的方法了。若是註釋掉testB類中的test()方法,則會報錯。由於程序沒法區分出你要使用的是哪個trait中的test()方法。咱們可使用insteadof來指定要使用的方法調用哪個trait。微信
class testE{ use B, C { B::test insteadOf C; C::test as testC; } } $e = new testE(); $e->test(); // This is trait B! $e->testC(); // This is trait C!
固然,現實開發中仍是儘可能規範方法名,不要出現這種重複狀況。另外,若是子類引用了trait,而父類又定義了一樣的方法呢?固然仍是調用父類所繼承來的方法。trait的優先級是低於普通的類繼承的。app
trait D{ function test(){ echo 'This is trait D!'; } } class parentD{ function test(){ echo 'This is class parentD'; } } class testD extends parentD{ use D; } $d = new testD(); $d->test(); // This is trait D!
最後,trait中也是能夠定義抽象方法的。這個抽象方法是引用這個trait的類所必須實現的方法,和抽象類中的抽象方法效果一致。oop
trait F{ function test(){ echo 'This is trait F!'; } abstract function show(); } class testF{ use F; function show(){ echo 'This is class testF!'; } } $f = new testF(); $f->test(); $f->show();
trait真的是PHP是很是靈活的一個功能。固然,越是靈活的東西越須要咱們去弄明白它的一些使用規則,這樣才能避免一些不可預見的錯誤。學習
關注公衆號:【硬核項目經理】獲取最新文章this
添加微信/QQ好友:【xiaoyuezigonggong/149844827】免費得PHP、項目管理學習資料.net
知乎、公衆號、抖音、頭條搜索【硬核項目經理】
B站ID:482780532