記得有一道面試題問php是否支持多繼承?
答案:不可以,只支持單繼承。
如何實現多繼承呢?
答案:可以使用 interface 或 trait 實現 。
為什么會想到這個問題,因為想到如果類繼承多個接口,然后他們之間還有相同的屬性和方法會引用誰的方法或屬性,誰又會被覆蓋?
總結:
1.使用 interface 聲明類不能被實例化,並且屬性必須是常量,方法不能有方法體
2.trait 聲明的類不能被實例化,由use引入,會覆蓋父類的相同屬性及方法,如果有多個use,那么按順序下面的覆蓋最上面的相同的屬性及方法
接口是什么?
使用接口(interface),可以指定某個類必須實現哪些方法,但不需要定義這些方法的具體內容。
接口是通過 interface 關鍵字來定義的,就像定義一個標准的類一樣,但其中定義所有的方法都是空的。
接口中定義的所有方法都必須是公有,這是接口的特性。
trait是什么?
看上去既像類又像接口,其實都不是,Trait可以看做類的部分實現,可以混入一個或多個現有的PHP類中,
其作用有兩個:表明類可以做什么;提供模塊化實現。
Trait是一種代碼復用技術,為PHP的單繼承限制提供了一套靈活的代碼復用機制。
參考文章:
接口:https://www.cnblogs.com/minigrasshopper/p/7754512.html
trait:https://blog.csdn.net/lemony521/article/details/78322652
php接口interface的使用
接口是什么?
使用接口(interface),可以指定某個類必須實現哪些方法,但不需要定義這些方法的具體內容。
接口是通過 interface 關鍵字來定義的,就像定義一個標准的類一樣,但其中定義所有的方法都是空的。
接口中定義的所有方法都必須是公有,這是接口的特性。
什么時候用接口?
1、定規范,保持統一性;
2、多個平級的類需要去實現同樣的方法,只是實現方式不一樣
接口使用規范
- 接口不能實例化
- 接口的屬性必須是常量
- 接口的方法必須是public【默認public】,且不能有函數體
- 類必須實現接口的所有方法
- 一個類可以同時實現多個接口,用逗號隔開
- 接口可以繼承接口【用的少】
interface usb{ const brand = 'siemens'; // 接口的屬性必須是常量 public function connect(); // 接口的方法必須是public【默認public】,且不能有函數體 } // new usb(); // 接口不能實例化 // 類實現接口 class Android implements usb{ public function connect(){ // 類必須實現接口的所有方法 echo '實現接口的connect方法'; } } interface usbA{ public function connect(); } interface usbB{ public function contact(); } // 類可以同時實現多個接口 class mi implements usbA,usbB{ public function connect(){ } public function contact(){ } }
php中trait的使用
1、php中的trait是啥?
看上去既像類又像接口,其實都不是,Trait可以看做類的部分實現,可以混入一個或多個現有的PHP類中,其作用有兩個:表明類可以做什么;提供模塊化實現。Trait是一種代碼復用技術,為PHP的單繼承限制提供了一套靈活的代碼復用機制。
2、PHP版本要求:
php5.4開始引入trait,其目的就是在於減少代碼的重復,增加代碼的復用性。
3、trait的使用場景:
試想這樣一種情況,當有一個方法需要在很多的類中使用時,該怎么處理?
通常一般的處理方式會是,寫一個基礎類,在基類中實現這個方法,然后所有類都繼承這個基類。
這是一種處理方法,但不是最好的處理方式。通常采用繼承的情況是:幾個類具有很大的相似性。比如人作為一個基類,學生、工人、等繼承“人”這個基類來擴展。
由此,trait的作用就出來了,trait 可以在多個類中使用。
4、trait如何使用:
引用PHP手冊中的例子:
例子一
<?php trait ezcReflectionReturnInfo { function getReturnType() { /*1*/ } function getReturnDescription() { /*2*/ } } class ezcReflectionMethod extends ReflectionMethod { use ezcReflectionReturnInfo; /* ... */ } class ezcReflectionFunction extends ReflectionFunction { use ezcReflectionReturnInfo; /* ... */ } ?>
1、先聲明一個trait;
2、在類中使用use將該trait引入。
是不是非常簡單(手動逃)?需要注意的是trait的優先級。
5、trait的優先級
(敲黑板)從基類繼承的成員會被 trait 插入的成員所覆蓋。優先順序是來自當前類的成員覆蓋了 trait 的方法,而 trait 則覆蓋了被繼承的方法。
優先級:自身方法>trait的方法>繼承的方法(就是這樣子的。)
看例子
<?php trait HelloWorld { public function sayHello() { echo 'Hello World!'; } } class TheWorldIsNotEnough { use HelloWorld; public function sayHello() { echo 'Hello Universe!'; } } $o = new TheWorldIsNotEnough(); $o->sayHello();//輸出是 Hello Universe! ?>
還有一點需要注意的是:多個trait的使用。
<?php trait Hello { public function sayHello() { echo 'Hello '; } } trait World { public function sayWorld() { echo 'World'; } } class MyHelloWorld { use Hello, World; public function sayExclamationMark() { echo '!'; } } $o = new MyHelloWorld(); $o->sayHello(); $o->sayWorld(); $o->sayExclamationMark(); ?>
總結:Trait是一種代碼復用技術,為PHP的單繼承限制提供了一套靈活的代碼復用機制。
附:以上是trait的基本使用,其還有幾個比較高級的使用注意點,詳細的使用方法可以參見PHP手冊,這里給出鏈接:PHP手冊中trait的使用!以上內容中的例子都是來自於該手冊。
參考:
1. http://php.net/manual/zh/language.oop5.traits.php
2. http://laravelacademy.org/post/4281.html
3. http://www.jianshu.com/p/47f0cdbe9b2c