Heim >Backend-Entwicklung >PHP-Tutorial >PHP中实现代码复用的Trait方法的一些特性

PHP中实现代码复用的Trait方法的一些特性

WBOY
WBOYOriginal
2016-06-20 12:31:591051Durchsuche

在整理“PHP基础”这篇文档时,看到了trait方法,感觉比较陌生,所以今天上午用两个小时的时间,查阅测试了trait方法的一些特性及用法,整理发布了这篇博客。

Trait 是 PHP5.4 中的新特性,是 PHP 多重继承的一种解决方案。例如,需要同时继承两个 Abstract Class, 这将会是件很麻烦的事情,Trait 就是为了解决这个问题。


trait的简单使用

trait使用前需要先定义,trait的定义方式和类的定义方式差不多

trait first_trait {
    function first_method() { /* Code Here */ }
    function second_method() { /* Code Here */ }

}

同时,如果要在 Class 中使用该 Trait,那么需要使用 use 关键字

class first_class {
    // 注意这行,声明使用 first_trait
    use first_trait;
}
$obj = new first_class();// Executing the method from trait
$obj->first_method(); // valid

$obj->second_method(); // valid

我们在类中可以直接声明使用被定义好的trait,之后在类被实例化后,就可以使用调用对象中方法的方式直接调用trait中的方法。无疑这增加了很多扩展性,因为类的继承是单一的,而trait的使用却不是单一的。


一个class中使用多个trait
在同一个 Class 中可以使用多个 Trait,多个 Trait之间通过逗号分隔。

例如:

trait first_trait
{
    function first_method() { echo "method1"; }
}
trait second_trait {
    function second_method() { echo "method2"; }
}
class first_class {
    // now using more than one trait
    use first_trait, second_trait;
}
$obj= new first_class();
// Valid
$obj->first_method(); // Print : method1
// Valid

$obj->second_method(); // Print : method2


Trait 冲突

多个 Trait 之间同时使用难免会冲突,例如,如果两个 trait 都插入了一个同名的方法,如果没有明确解决冲突将会产生一个致命错误。

为了解决多个 trait 在同一个类中的命名冲突,可以使用关键字:insteadof 以及 as ;
insteadof 操作符用来明确指定使用哪一个冲突方法;
as 操作符用来给冲突的方法重新命名并引入。


trait first_trait {
    function first_function() { 
        echo "From First Trait";
    }
}
trait second_trait {
    // 这里的名称和 first_trait 一样,会有冲突
    function first_function() { 
        echo "From Second Trait";
    }
}
class first_class {
    use first_trait, second_trait {
        // 在这里声明使用 first_trait 的 first_function 替换
        // second_trait 中声明的
        first_trait::first_function insteadof second_trait;

        second_trait::first_function as second_trait;

    }
}  


Trait 的抽象方法

我们可以在Trait 中声明抽象方法,这样,使用它的 Class 必须实现它。

trait first_trait {
    function first_method() { echo "method1"; }
    abstract public function second_method();// 这里可以加入修饰符,说明调用类必须实现它
}
class first_method {
    use first_trait;
    function second_method() {
        /* Code Here */
    }

}


Trait优先级

PHP手册说法:优先顺序是当前类的方法覆盖了 trait 的方法,而 trait 覆盖了被继承的方法。

即优先级:当前类方法 > trait方法 > 被继承的方法。

如果在不同优先级中定义了一个同名的方法,那么优先级高的方法将覆盖优先级低的方法;

两个trait属于同一优先级,若出现同名方法,则会报致命错误,具体参见上面。

class Base {
    public function sayHello() {
        echo 'Hello ';
    }
}
trait SayWorld {
    public function sayHello() {
        parent::sayHello();
        echo 'World!';
    }
}
class MyHelloWorld extends Base {
    use SayWorld;
}
$o = new MyHelloWorld();
$o->sayHello();

结果会输出:

Hello World!


Trait访问权限控制

与类相似的,我们在定义trait的方法时,也可以指定其访问权限(public、protected、private);

我们还可以使用as来修改方法的访问权限。

trait HelloWorld {
    public function sayHello() {
        echo 'Hello World!';
    }
}
// 修改 sayHello 的访问控制
class MyClass1 {
    use HelloWorld { sayHello as protected; }
}
// 给方法一个改变了访问控制的别名
// 原版 sayHello 的访问控制则没有发生变化
class MyClass2 {
    use HelloWorld { sayHello as private myPrivateHello; }

}

Trait 之间的嵌套

使用trait时,可以将不同的trait组成新的trait,这样使得代码的复用率变得更高,当然对程序的设计要求也更高。

trait Hello {

    public function sayHello() {

        echo 'Hello ';

    }}

 

trait World {

    public function sayWorld() {

        echo 'World!';

    }}

 

trait HelloWorld {

    use Hello, World;

}

 

class MyHelloWorld {

    use HelloWorld;

}

 

$o = new MyHelloWorld();

$o->sayHello();

$o->sayWorld();

 //输出  Hello World!


Trait 的静态成员

与类相似,静态成员分静态方法和静态属性,二者都是以static关键字来进行声明的。

trait Counter {

    public function inc() {

        static $c = 0;

        $c = $c + 1;

        echo "$c\n";

    }

}

trait StaticExample {

    public static function doSomething() {

        return 'Doing something';

    }

}


Trait 属性

在trait中定义了属性后,类的实例可以像访问自身的属性一样访问它。而且可以给它设置访问权限,就差不能实例化了……

需要注意的是,如果 trait 定义了一个属性,那类将不能定义同样名称的属性,否则会产生一个错误。如果该属性在类中的定义与在 trait 中的定义兼容(同样的可见性和初始值)则错误的级别是 E_STRICT,否则是一个致命错误。

trait PropertiesTrait {

    public $x = 1;

}

 

class PropertiesExample {

    use PropertiesTrait;

}

 

$example = new PropertiesExample;

$example->x;


需要特别注意的几点

上面这些就是 Trait 基本的使用了,更详细的可以参k官方手册。

这里总结下注意的几 点:

    - Trait 会覆盖调用类继承的父类方法

    - Trait 无法如 Class 一样使用 new 实例化

    - 单个 Trait 可由多个 Trait 组成

    - 在单个 Class 中,可以使用多个 Trait

    - Trait 支持修饰词(modifiers),例如 final、static、abstract

    - 我们能使用 insteadof 以及 as 操作符解决 Trait 之间的冲突


参K文献

    - http://php.net/manual/zh/language.oop5.traits.php

    - https://www.doubear.com/article/raityongfaxuexibiji.html

    - http://www.kuqin.com/web/20111119/315048.html

    - http://www.jb51.net/article/61260.htm


最后

欢迎关注我的个人微信“华伟君”,与我一起学习探讨PHP,通往技术大牛的路上,我们结伴同行!

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn