溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

PHP中trait的使用和同時(shí)引入多個(gè)trait時(shí)同名方法沖突怎么處理

發(fā)布時(shí)間:2023-04-26 16:29:40 來源:億速云 閱讀:157 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要講解了“PHP中trait的使用和同時(shí)引入多個(gè)trait時(shí)同名方法沖突怎么處理”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“PHP中trait的使用和同時(shí)引入多個(gè)trait時(shí)同名方法沖突怎么處理”吧!

PHP的類是單一繼承模式,也就是每個(gè)類只能繼承一個(gè)父類(基類)。

但有時(shí)需要引入更多通用(共用)的方法,同時(shí)這些方法又不適合集成到基類。

那么這時(shí),就需要使用其他方法來引入這些方法。其中trait,就是方法之一。

trait是PHP5.4之后出現(xiàn)的一種代碼復(fù)用方法,形式和Class非常相似,同時(shí)可以隨意組合任意引入。

trait一般在當(dāng)前類和其同父類(基類)的其他類都需要使用相同方法時(shí),而其父類(基類)又要盡量避免出現(xiàn)這些方法時(shí)使用。

甚至有時(shí)可能其他關(guān)聯(lián)不是特別大的類(分別繼承不同的父類)也可能會(huì)使用共同的方法,也可以使用trait的方法。

盡量通俗一點(diǎn)的說一下trait:

trait像類,但不是類,不可以直接使用new關(guān)鍵字創(chuàng)建對(duì)象;簡(jiǎn)單理解是用類的形式,封裝一大堆通用(共用)的方法,供其他類引用。

trait和use搭配使用。定義好trait后,“use trait定義的名字;”,就可以直接使用里邊定義的一切了,是不是很簡(jiǎn)單?很方便?

現(xiàn)在知道了trait,接下來就通過代碼實(shí)例,演示一下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é)果截圖:

PHP中trait的使用和同時(shí)引入多個(gè)trait時(shí)同名方法沖突怎么處理

上邊的這個(gè)例子,子類直接extentds父類,然后又在類內(nèi)use了trait。這樣當(dāng)前類(子類)就擁有了這三個(gè)的全部方法。

子類的sub方法,父類的parent方法,trait的test方法,在子類內(nèi)都可以直接調(diào)用使用。

最基礎(chǔ)的使用就這些,看起來是不是也不算難?甚至感覺挺簡(jiǎn)單的?

那么我們進(jìn)一步思考一下,類的“繼承”難免會(huì)出現(xiàn)同名方法,那么這三個(gè)里邊如果有同名方法,最終會(huì)保留哪個(gè)?誰的方法會(huì)被覆蓋呢?

二、當(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é)果截圖:

PHP中trait的使用和同時(shí)引入多個(gè)trait時(shí)同名方法沖突怎么處理

上邊這段例子的結(jié)果很明顯的發(fā)現(xiàn),最終當(dāng)前類(子類)的方法被調(diào)用了,也就是三個(gè)里邊都有同名方法時(shí),當(dāng)前類的方法優(yōu)先。

接下來,注釋(刪除)當(dāng)前類的lookClassName()方法。

PHP中trait的使用和同時(shí)引入多個(gè)trait時(shí)同名方法沖突怎么處理

看上邊截圖,很明顯了,當(dāng)子類(當(dāng)前類)沒有同名方法,只有父類(基類)和trait中的方法同名時(shí),trait中的方法優(yōu)先。

結(jié)論:當(dāng)前類(子類)、trait和父類(基類)中有同名方法時(shí)“子類高于trait高于父類”。子類的方法會(huì)覆蓋trait中的方法,而trait的方法會(huì)覆蓋父類的方法。

前邊有提到,trait可以隨意組合,隨意引用,那么是不是可以同時(shí)引入多個(gè)trait呢?是。在一個(gè)類內(nèi),可以同時(shí)use多個(gè)trait。

三、類內(nèi)同時(shí)引入多個(gè)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é)果截圖:

PHP中trait的使用和同時(shí)引入多個(gè)trait時(shí)同名方法沖突怎么處理

當(dāng)需要同時(shí)引入多個(gè)trait時(shí),只要use trait1, trait2, trait3,在use后邊跟多個(gè)trait名字即可,多個(gè)trait之間用逗號(hào)分隔。

當(dāng)然,也可以分開寫,每次use一個(gè)trait進(jìn)來。

此時(shí)又有新的問題產(chǎn)生了,如果引入的多個(gè)trait都有同名的方法,那么又會(huì)是誰優(yōu)先?誰又被覆蓋呢?

四、引入多個(gè)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é)果截圖:

PHP中trait的使用和同時(shí)引入多個(gè)trait時(shí)同名方法沖突怎么處理

說明(上邊的源碼和結(jié)果是解沖突之后的):

當(dāng)子類沒有(注釋或者刪除)lookClassName()方法時(shí),調(diào)用lookClassName方法,則會(huì)調(diào)用trait中的方法,因?yàn)槿齻€(gè)trait中都有同名方法,此時(shí)就會(huì)發(fā)生致命錯(cuò)誤(沖突)。

報(bào)下邊(看截圖)的語法錯(cuò)誤

PHP中trait的使用和同時(shí)引入多個(gè)trait時(shí)同名方法沖突怎么處理

此時(shí),就需要解沖突。

解沖突,就需要使用到insteadof關(guān)鍵字,含義是“代替”,就是用哪個(gè)代替哪個(gè)。

 use traitTest, traitTest2, traitTest3 {
        traitTest2::lookClassName insteadof traitTest;// traitTest2代替了traitTest
        traitTest3::lookClassName insteadof traitTest2;// traitTest3代替了traitTest2
    }

解引入多個(gè)trait多個(gè)重名方法沖突時(shí),需要在引入時(shí)使用insteadof關(guān)鍵字,逐一說明哪個(gè)trait的方法代替了哪個(gè)trait的(看上邊引入代碼的注釋)。

根據(jù)上邊引入的代碼,是traitTest2的lookClassName代替了traitTest的,然后traitTest3的代替了traitTest2的。

因此,最終輸出結(jié)果時(shí),調(diào)用lookClassName(),輸出的就是traitTest3的內(nèi)容(輸出結(jié)果看上邊最近的“代碼和結(jié)果截圖”)。

當(dāng)然,也可以換個(gè)寫法:

 use traitTest, traitTest2, traitTest3 {
        traitTest2::lookClassName insteadof traitTest3;// traitTest2代替了traitTest3
        traitTest3::lookClassName insteadof traitTest2;// traitTest3代替了traitTest2
    }

這個(gè)寫法呢,是traitTest2和traitTest3互相代替了,那么此時(shí)反而沒有traitTest什么事了。這個(gè)時(shí)候,再調(diào)用lookClassName()方法,輸出的就是traitTest的lookClassName()方法的內(nèi)容。

代碼和結(jié)果截圖:

PHP中trait的使用和同時(shí)引入多個(gè)trait時(shí)同名方法沖突怎么處理

如圖,當(dāng)traitTest2和traitTest3互相代替后,直接輸出了traitTest的內(nèi)容。

到這基本就該結(jié)束了,但,有個(gè)特殊情況需要考慮一下。

我們之所以會(huì)引入多個(gè)trait,說明這幾個(gè)trait里都有想使用的方法,那么非常巧合,其中同名方法正好又都想使用,被代替的方法還能使用么?

五、當(dāng)引入多個(gè)trait,同名方法解沖突后,同時(shí)使用所有沖突的同名方法

解決:我們需要使用到另一個(gè)關(guān)鍵字“as”,此關(guān)鍵字的功能,簡(jiǎn)單理解就是給方法取一個(gè)別名。

代碼:

// 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é)果截圖:

PHP中trait的使用和同時(shí)引入多個(gè)trait時(shí)同名方法沖突怎么處理

根據(jù)上圖,就可以看出,當(dāng)trait2和trait3互相代替,然后同名方法另起別名后,三個(gè)trait的同名方法,不再?zèng)_突,并且可以分別調(diào)用各自原本同名的方法。

感謝各位的閱讀,以上就是“PHP中trait的使用和同時(shí)引入多個(gè)trait時(shí)同名方法沖突怎么處理”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對(duì)PHP中trait的使用和同時(shí)引入多個(gè)trait時(shí)同名方法沖突怎么處理這一問題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI