PHP之Trait詳解

php從以前到現在一直都是單繼承的語言,無法同時從兩個基類中繼承屬性和方法,爲了解決這個問題,php出了Trait這個特性

用法:通過在類中使用use 關鍵字,聲明要組合的Trait名稱,具體的Trait的聲明使用Trait關鍵詞,Trait不能實例化

如下代碼實例:

 

<?php
trait Dog{
    public $name="dog";
    public function bark(){
        echo "This is dog";
    }
}
class Animal{
    public function eat(){
        echo "This is animal eat";
    }
}
class Cat extends Animal{
    use Dog;
    public function drive(){
        echo "This is cat drive";
    }
}
$cat = new Cat();
$cat->drive();
echo "<br/>";
$cat->eat();
echo "<br/>";
$cat->bark();
?>

將會如下輸出

Paste_Image.png

再測試Trait、基類和本類對同名屬性或方法的處理,如下代碼

 

<?php
trait Dog{
    public $name="dog";
    public function drive(){
        echo "This is dog drive";
    }
    public function eat(){
        echo "This is dog eat";
    }
}

class Animal{
    public function drive(){
        echo "This is animal drive";
    }
    public function eat(){
        echo "This is animal eat";
    }
}

class Cat extends Animal{
    use Dog;
    public function drive(){
        echo "This is cat drive";
    }
}
$cat = new Cat();
$cat->drive();
echo "<br/>";
$cat->eat();

?>

如下顯示

Paste_Image.png

所以:Trait中的方法會覆蓋 基類中的同名方法,而本類會覆蓋Trait中同名方法
注意點:當trait定義了屬性後,類就不能定義同樣名稱的屬性,否則會產生 fatal error,除非是設置成相同可見度、相同默認值。不過在php7之前,即使這樣設置,還是會產生E_STRICT 的提醒

一個類可以組合多個Trait,通過逗號相隔,如下

use trait1,trait2

當不同的trait中,卻有着同名的方法或屬性,會產生衝突,可以使用insteadof或 as進行解決,insteadof 是進行替代,而as是給它取別名
如下實例:

 

<?php
trait trait1{
    public function eat(){
        echo "This is trait1 eat";
    }
    public function drive(){
        echo "This is trait1 drive";
    }
}
trait trait2{
    public function eat(){
        echo "This is trait2 eat";
    }
    public function drive(){
        echo "This is trait2 drive";
    }
}
class cat{
    use trait1,trait2{
        trait1::eat insteadof trait2;
        trait1::drive insteadof trait2;
    }
}
class dog{
    use trait1,trait2{
        trait1::eat insteadof trait2;
        trait1::drive insteadof trait2;
        trait2::eat as eaten;
        trait2::drive as driven;
    }
}
$cat = new cat();
$cat->eat();
echo "<br/>";
$cat->drive();
echo "<br/>";
echo "<br/>";
echo "<br/>";
$dog = new dog();
$dog->eat();
echo "<br/>";
$dog->drive();
echo "<br/>";
$dog->eaten();
echo "<br/>";
$dog->driven();
?>

輸出如下

Paste_Image.png

as 還可以修改方法的訪問控制

 

<?php
trait Animal{
    public function eat(){
        echo "This is Animal eat";
    }
}

class Dog{
    use Animal{
        eat as protected;
    }
}
class Cat{
    use Animal{
        Animal::eat as private eaten;
    }
}
$dog = new Dog();
$dog->eat();//報錯,因爲已經把eat改成了保護

$cat = new Cat();
$cat->eat();//正常運行,不會修改原先的訪問控制
$cat->eaten();//報錯,已經改成了私有的訪問控制
?>

Trait也可以互相組合,還可以使用抽象方法,靜態屬性,靜態方法等,實例如下

 

<?php
trait Cat{
    public function eat(){
        echo "This is Cat eat";
    }
}

trait Dog{
    use Cat;
    public function drive(){
        echo "This is Dog drive";
    }
    abstract public function getName();
    
    public function test(){
        static $num=0;
        $num++;
        echo $num;
    }
    
    public static function say(){
        echo "This is Dog say";
    }
}
class animal{
    use Dog;
    public function getName(){
        echo "This is animal name";
    }
}

$animal = new animal();
$animal->getName();
echo "<br/>";
$animal->eat();
echo "<br/>";
$animal->drive();
echo "<br/>";
$animal::say();
echo "<br/>";
$animal->test();
echo "<br/>";
$animal->test();
?>

輸出如下

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章