php中的trait

    xiaoxiao2025-05-16  9

    PHP5.4起,新增一种新的代码复用的方法,trait。PHP与JAVA类似,与C++不同,是一种单继承语言,PHP引入trait,可以减少单继承语言的限制,使开发人员能够自由地在不同的层次结构内独立的类中复用method。Trait和Class组合的语义定义了一种减少复杂性的方式,避免传统多继承和Mixin类相关经典问题,例如C++中的钻石危机(也称棱形问题): 如果在一个子类继承的多个父类中拥有相同名字的实例变量,子类在引用该变量时将产生歧义,无法判断应该使用哪个父类的变量。如果在一个子类继承的多个父类中拥有相同方法,子类中有没有覆盖该方法,那么调用该方法时将产生歧义,无法判断应该调用哪个父类的方法。

    Trait和Class相似,但仅仅旨在用细粒度和一致的方式来组合功能。无法通过trait自身来实例化。它为传统继承增加了水平特性的组合。

    Trait使用case

    <?php trait Mytrait { public $a = 100; public function test1() { echo "Mytrait Test1\n"; } public static function test2() { echo "Mytrait Test2\n"; } } class TraitExample1 { use Mytrait; } $obj = new TraitExample1(); $obj->test1(); TraitExample1::test2(); echo $obj->a . "\n";

    优先级

    对于方法: 当前类的成员>trait的方法>从基类继承的成员

    对于变量: 当前类和基类中不能存在与trait重名的变量,否则会抛出fatal error。

    <?php trait Mytrait { public $a = 100; public function test1() { echo "Mytrait Test1\n"; } public static function test2() { echo "Mytrait Test2\n"; } } class Base { public function test1() { echo "Base Test1\n"; } public static function test2() { echo "Base Test2\n"; } public static function test3() { echo "Base Test3\n"; } } class TraitExample1 extends Base{ use Mytrait; public function test1() { echo "TraitExample1 Test1\n"; } } $obj = new TraitExample1(); $obj->test1(); TraitExample1::test2(); $obj->test3(); echo $obj->a . "\n";

    多个trait

    通过逗号分隔,在use声明中列出多个trait,可以都插入到一个类中。

    如果两个trait都插入了一个同名的方法,如果没有明确解决冲突将会产生一个致命错误。为了解决多个trait在同一个类中的命名冲突,需要使用insteadof操作符来明确指定使用冲突方法中的哪一个。或者使用as操作符,可以将其中一个冲突的方法以另一个名称来引入。

    同样的,如果两个trait有同名的变量,如果该属性的定义兼容(同样的可见性及初始值)错误级别是E_STRICT,否则错误级别是E_ERROR.

    且不能通过解决方法冲突的方式解决。

    <?php trait Mytrait { public $a = 100; public function test1() { echo "Mytrait Test1\n"; } public static function test2() { echo "Mytrait Test2\n"; } } trait Mytrait2 { public function test1() { echo "Mytrait2 Test1\n"; } public static function test2() { echo "Mytrait2 Test2\n"; } } class TraitExample1{ use Mytrait,Mytrait2 { Mytrait::test1 insteadof Mytrait2; Mytrait2::test2 insteadof Mytrait; Mytrait::test2 as test2_2; } } $obj = new TraitExample1(); $obj->test1(); TraitExample1::test2(); $obj->test2_2(); echo $obj->a . "\n";

    修改方法的访问控制

    <?php trait Mytrait { public function test1() { echo "Mytrait Test1\n"; } public function test2() { echo "Mytrait Test2\n"; } private function test3() { echo "Mytrait Test3\n"; } } class TraitExample1{ use Mytrait { test1 as protected; test2 as public test2_1; test3 as public; } } 指定alias会创建一个新的方法,原方法访问控制权限不变。不仅可以缩小权限的方向修改(public->private),也可以向放大权限的修改(private->public).

    从trait来组成trait

    与Class中使用多个trait类似,trait也可以使用多个trait。冲突的解决方式也一致。

    <?php trait Mytrait { public function test1() { echo "Mytrait Test1\n"; } public function test2() { echo "Mytrait Test2\n"; } } trait Mytrait2 { public function test1() { echo "Mytrait2 Test1\n"; } public function test2() { echo "Mytrait2 Test2\n"; } } trait Mytrait3 { use Mytrait, Mytrait2 { Mytrait::test1 insteadof Mytrait2; Mytrait2::test2 insteadof Mytrait; } } class TraitExample1{ use Mytrait3; } $obj = new TraitExample1(); $obj->test1(); $obj->test2();

    Trait的抽象成员

    为了对使用的类施加强制要求,trait支持抽象方法的使用。

    Trait的静态成员

    trait可以定义静态变量,静态方法。

    <?php trait Mytrait { public static $in = 10; public function test1() { echo "Mytrait Test1\n"; } public function test2() { echo "Mytrait Test2\n"; } } trait Mytrait2 { public function test1() { echo "Mytrait2 Test1\n"; } public static function test2() { echo "Mytrait2 Test2\n"; } } trait Mytrait3 { use Mytrait, Mytrait2 { Mytrait::test1 insteadof Mytrait2; Mytrait::test2 insteadof Mytrait2; Mytrait2::test2 as test3; } } class TraitExample1{ use Mytrait3; } $obj = new TraitExample1(); $obj->test1(); $obj->test2(); TraitExample1::test3(); [root@hadoop1 php]# cat trait2.php <?php trait Mytrait { public static $in = 10; public function test1() { echo "Mytrait Test1\n"; } public function test2() { echo "Mytrait Test2\n"; } } trait Mytrait2 { public function test1() { echo "Mytrait2 Test1\n"; } public static function test2() { echo "Mytrait2 Test2\n"; } } trait Mytrait3 { use Mytrait, Mytrait2 { Mytrait::test1 insteadof Mytrait2; Mytrait::test2 insteadof Mytrait2; Mytrait2::test2 as test3; } } class TraitExample1{ use Mytrait3; } $obj = new TraitExample1(); $obj->test1(); $obj->test2(); TraitExample1::test3(); echo TraitExample1::$in;

    转载请注明原文地址: https://ju.6miu.com/read-1298946.html
    最新回复(0)