PHP中trait的使用和同時引入多個trait時同名方法沖突的處理方法
PHP的類是單一繼承模式,也就是每個類只能繼承一個父類(基類)。
但有時需要引入更多通用(共用)的方法,同時這些方法又不適合集成到基類。
那么這時,就需要使用其他方法來引入這些方法。其中trait,就是方法之一。
trait是PHP5.4之后出現(xiàn)的一種代碼復(fù)用方法,形式和Class非常相似,同時可以隨意組合任意引入。
trait一般在當(dāng)前類和其同父類(基類)的其他類都需要使用相同方法時,而其父類(基類)又要盡量避免出現(xiàn)這些方法時使用。
甚至有時可能其他關(guān)聯(lián)不是特別大的類(分別繼承不同的父類)也可能會使用共同的方法,也可以使用trait的方法。
盡量通俗一點的說一下trait:
trait像類,但不是類,不可以直接使用new關(guān)鍵字創(chuàng)建對象;簡單理解是用類的形式,封裝一大堆通用(共用)的方法,供其他類引用。
trait和use搭配使用。定義好trait后,“use trait定義的名字;”,就可以直接使用里邊定義的一切了,是不是很簡單?很方便?
現(xiàn)在知道了trait,接下來就通過代碼實例,演示一下trait的具體使用和一些小情況。
一、trait的使用
代碼:
// trait trait traitTest { public function test() { echo "trait test...\n"; } } // 父類 class ParentClass { public function parent() { echo "parent...\n"; } } // 子類 class SubClass extends ParentClass { use traitTest; public function sub() { echo "sub...\n"; } } $obj = new SubClass; $obj->sub();// 調(diào)用子類方法 $obj->parent();// 調(diào)用父類的方法 $obj->test();// 調(diào)用trait里的方法
代碼和結(jié)果截圖:
上邊的這個例子,子類直接extentds父類,然后又在類內(nèi)use了trait。這樣當(dāng)前類(子類)就擁有了這三個的全部方法。
子類的sub方法,父類的parent方法,trait的test方法,在子類內(nèi)都可以直接調(diào)用使用。
最基礎(chǔ)的使用就這些,看起來是不是也不算難?甚至感覺挺簡單的?
那么我們進一步思考一下,類的“繼承”難免會出現(xiàn)同名方法,那么這三個里邊如果有同名方法,最終會保留哪個?誰的方法會被覆蓋呢?
二、當(dāng)父類、子類和trait的方法重名
代碼:
// trait trait traitTest { public function test() { echo "trait test...\n"; } public function lookClassName() { echo "trait here\n"; echo __CLASS__ . "\n"; } } // 父類 class ParentClass { public function parent() { echo "parent...\n"; } public function lookClassName() { echo __CLASS__ . "\n"; } } // 子類 class SubClass extends ParentClass { use traitTest; public function sub() { echo "sub...\n"; } public function lookClassName() { echo __CLASS__ . "\n"; } } $obj = new SubClass; $obj->sub();// 調(diào)用子類方法 $obj->parent();// 調(diào)用父類的方法 $obj->test();// 調(diào)用trait里的方法 $obj->lookClassName();// 調(diào)用同名方法
代碼和結(jié)果截圖:
上邊這段例子的結(jié)果很明顯的發(fā)現(xiàn),最終當(dāng)前類(子類)的方法被調(diào)用了,也就是三個里邊都有同名方法時,當(dāng)前類的方法優(yōu)先。
接下來,注釋(刪除)當(dāng)前類的lookClassName()方法。
看上邊截圖,很明顯了,當(dāng)子類(當(dāng)前類)沒有同名方法,只有父類(基類)和trait中的方法同名時,trait中的方法優(yōu)先。
結(jié)論:當(dāng)前類(子類)、trait和父類(基類)中有同名方法時“子類高于trait高于父類”。子類的方法會覆蓋trait中的方法,而trait的方法會覆蓋父類的方法。
前邊有提到,trait可以隨意組合,隨意引用,那么是不是可以同時引入多個trait呢?是。在一個類內(nèi),可以同時use多個trait。
三、類內(nèi)同時引入多個trait
// trait trait traitTest { public function test() { echo "trait test...\n"; } public function lookClassName() { echo "trait here\n"; echo __CLASS__ . "\n"; } } trait traitTest2 { public function test2() { echo "trait2 test...\n"; } public function lookClassName() { echo "trait2 here\n"; echo __CLASS__ . "\n"; } } trait traitTest3 { public function test3() { echo "trait3 test...\n"; } public function lookClassName() { echo "trait3 here\n"; echo __CLASS__ . "\n"; } } // 父類 class ParentClass { public function parent() { echo "parent...\n"; } public function lookClassName() { echo __CLASS__ . "\n"; } } // 子類 class SubClass extends ParentClass { use traitTest; use traitTest2, traitTest3; public function sub() { echo "sub...\n"; } public function lookClassName() { echo __CLASS__ . "\n"; } } $obj = new SubClass; $obj->sub();// 調(diào)用子類方法 $obj->parent();// 調(diào)用父類的方法 $obj->test();// 調(diào)用trait里的方法 $obj->test2();// 調(diào)用trait2里的方法 $obj->test3();// 調(diào)用trait3里的方法 $obj->lookClassName();// 調(diào)用同名方法
代碼和結(jié)果截圖:
當(dāng)需要同時引入多個trait時,只要use trait1, trait2, trait3,在use后邊跟多個trait名字即可,多個trait之間用逗號分隔。
當(dāng)然,也可以分開寫,每次use一個trait進來。
此時又有新的問題產(chǎn)生了,如果引入的多個trait都有同名的方法,那么又會是誰優(yōu)先?誰又被覆蓋呢?
四、引入多個trait有同名方法
代碼:
// trait trait traitTest { public function test() { echo "trait test...\n"; } public function lookClassName() { echo "trait here\n"; echo __CLASS__ . "\n"; } } trait traitTest2 { public function test2() { echo "trait2 test...\n"; } public function lookClassName() { echo "trait2 here\n"; echo __CLASS__ . "\n"; } } trait traitTest3 { public function test3() { echo "trait3 test...\n"; } public function lookClassName() { echo "trait3 here\n"; echo __CLASS__ . "\n"; } } // 父類 class ParentClass { public function parent() { echo "parent...\n"; } public function lookClassName() { echo __CLASS__ . "\n"; } } // 子類 class SubClass extends ParentClass { use traitTest, traitTest2, traitTest3 { traitTest2::lookClassName insteadof traitTest;// traitTest2代替了traitTest traitTest3::lookClassName insteadof traitTest2;// traitTest3代替了traitTest2 } public function sub() { echo "sub...\n"; } // public function lookClassName() { // echo __CLASS__ . "\n"; // } } $obj = new SubClass; $obj->sub();// 調(diào)用子類方法 $obj->parent();// 調(diào)用父類的方法 $obj->test();// 調(diào)用trait里的方法 $obj->test2();// 調(diào)用trait2里的方法 $obj->test3();// 調(diào)用trait3里的方法 $obj->lookClassName();// 調(diào)用同名方法
代碼和結(jié)果截圖:
說明(上邊的源碼和結(jié)果是解沖突之后的):
當(dāng)子類沒有(注釋或者刪除)lookClassName()方法時,調(diào)用lookClassName方法,則會調(diào)用trait中的方法,因為三個trait中都有同名方法,此時就會發(fā)生致命錯誤(沖突)。
報下邊(看截圖)的語法錯誤
此時,就需要解沖突。
解沖突,就需要使用到insteadof關(guān)鍵字,含義是“代替”,就是用哪個代替哪個。
use traitTest, traitTest2, traitTest3 { traitTest2::lookClassName insteadof traitTest;// traitTest2代替了traitTest traitTest3::lookClassName insteadof traitTest2;// traitTest3代替了traitTest2 }
解引入多個trait多個重名方法沖突時,需要在引入時使用insteadof關(guān)鍵字,逐一說明哪個trait的方法代替了哪個trait的(看上邊引入代碼的注釋)。
根據(jù)上邊引入的代碼,是traitTest2的lookClassName代替了traitTest的,然后traitTest3的代替了traitTest2的。
因此,最終輸出結(jié)果時,調(diào)用lookClassName(),輸出的就是traitTest3的內(nèi)容(輸出結(jié)果看上邊最近的“代碼和結(jié)果截圖”)。
當(dāng)然,也可以換個寫法:
use traitTest, traitTest2, traitTest3 { traitTest2::lookClassName insteadof traitTest3;// traitTest2代替了traitTest3 traitTest3::lookClassName insteadof traitTest2;// traitTest3代替了traitTest2 }
這個寫法呢,是traitTest2和traitTest3互相代替了,那么此時反而沒有traitTest什么事了。這個時候,再調(diào)用lookClassName()方法,輸出的就是traitTest的lookClassName()方法的內(nèi)容。
代碼和結(jié)果截圖:
如圖,當(dāng)traitTest2和traitTest3互相代替后,直接輸出了traitTest的內(nèi)容。
到這基本就該結(jié)束了,但,有個特殊情況需要考慮一下。
我們之所以會引入多個trait,說明這幾個trait里都有想使用的方法,那么非常巧合,其中同名方法正好又都想使用,被代替的方法還能使用么?
五、當(dāng)引入多個trait,同名方法解沖突后,同時使用所有沖突的同名方法
解決:我們需要使用到另一個關(guān)鍵字“as”,此關(guān)鍵字的功能,簡單理解就是給方法取一個別名。
代碼:
// trait trait traitTest { public function test() { echo "trait test...\n"; } public function lookClassName() { echo "trait here\n"; echo __CLASS__ . "\n"; } } trait traitTest2 { public function test2() { echo "trait2 test...\n"; } public function lookClassName() { echo "trait2 here\n"; echo __CLASS__ . "\n"; } } trait traitTest3 { public function test3() { echo "trait3 test...\n"; } public function lookClassName() { echo "trait3 here\n"; echo __CLASS__ . "\n"; } } // 父類 class ParentClass { public function parent() { echo "parent...\n"; } public function lookClassName() { echo __CLASS__ . "\n"; } } // 子類 class SubClass extends ParentClass { use traitTest, traitTest2, traitTest3 { traitTest2::lookClassName insteadof traitTest3;// traitTest2代替了traitTest3 traitTest3::lookClassName insteadof traitTest2;// traitTest3代替了traitTest2 traitTest2::lookClassName as lookClassName2;// traitTest2的lookClassName改別名lookClassName2 traitTest3::lookClassName as lookClassName3;// traitTest3的lookClassName改別名lookClassName3 } public function sub() { echo "sub...\n"; } // public function lookClassName() { // echo __CLASS__ . "\n"; // } } $obj = new SubClass; $obj->sub();// 調(diào)用子類方法 $obj->parent();// 調(diào)用父類的方法 $obj->test();// 調(diào)用trait里的方法 $obj->test2();// 調(diào)用trait2里的方法 $obj->test3();// 調(diào)用trait3里的方法 $obj->lookClassName();// 調(diào)用同名方法 $obj->lookClassName2();// 調(diào)用traitTest2更名后的同名方法 $obj->lookClassName3();// 調(diào)用traitTest3更名后的同名方法
代碼和結(jié)果截圖:
根據(jù)上圖,就可以看出,當(dāng)trait2和trait3互相代替,然后同名方法另起別名后,三個trait的同名方法,不再沖突,并且可以分別調(diào)用各自原本同名的方法。
到此要說的東西基本都說完了。算是對PHP的trait的一個小小的總結(jié),希望可以幫到需要的朋友。
到此這篇關(guān)于PHP中trait的使用和同時引入多個trait時同名方法沖突的處理的文章就介紹到這了,更多相關(guān)PHP中trait的使用內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
用Laravel輕松處理千萬級數(shù)據(jù)的方法實現(xiàn)
這篇文章主要介紹了用Laravel輕松處理千萬級數(shù)據(jù)的方法實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12Laravel學(xué)習(xí)教程之model validation的使用示例
這篇文章主要給大家介紹了關(guān)于Laravel學(xué)習(xí)教程之model validation使用的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起看看吧。2017-10-10visual studio code 調(diào)試php方法(圖文詳解)
本篇文章主要介紹了visual studio code 調(diào)試php方法(圖文詳解),具有一定的參考價值,有興趣的可以了解一下2017-09-09laravel框架之?dāng)?shù)據(jù)庫查出來的對象實現(xiàn)轉(zhuǎn)化為數(shù)組
今天小編就為大家分享一篇laravel框架之?dāng)?shù)據(jù)庫查出來的對象實現(xiàn)轉(zhuǎn)化為數(shù)組,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-10-10