自PHP 5.4.0起,PHP实现了一种代码服用的方法,称为trait。
Trait是为类似PHP的单继承语言而准备的一种代码复用机制。Trait为了减少单继承语言的限制,使开发人员能够自由地在不同层次结构内独立的类中服用 method。Trait和Class组合的语义定义了一种减少复杂性的方式,避免传统多继承和Mixin类相关典型问题。
Trait 和 Class 相似,但仅仅旨在用细粒度和一致的方式来组合功能。无法通过trait自身来实例化。它为传统继承增加了水平特性的组合;也就是说,应用的几个Class之间不需要继承。
通过PHP文档的定义和解释,我们可以理解为 trait 是某个class 的一部分,它可以和任意class合并,但 trait 不能实例化,同时也不能被其它类继承;trait 本质上与class没有任何的关系,当某个class引用trait时,才会与当前class合并。
实例如下:
<?php
trait SayWorld {
public function sayHello(){
echo 'world';
}
}
class MyHelloWorld {
//引入trait SayWorld 与 MyHelloWorld类合并
use SayWorld;
}
//实例化类
$hello = new MyHelloWorld;
//输出 world
$hello->sayHello();
需要注意的有以下几点:
- 优先级
如果当前类中的方法名称与引入的trait 中的方法相同时,则当前类中的方法会覆盖trait 中的方法;如果当前类继承某个基类,而基类中的方法名称与trait 中的方法名称相同时,则引入的trait 中的方法会覆盖基类中的方法。
如下所示:
<?php
trait HelloWorld {
public function sayHello() {
echo 'Hello World!';
}
}
class TheWorldIsNotEnough {
//引入HelloWorld与当前类TheWorldIsNotEnough合并
use HelloWorld;
//因为当前类与引入的HelloWorld中的方法重名,
//所以当前类TheWorldIsNotEnough中的方法覆盖了HelloWorld中的方法sayHello
public function sayHello() {
echo 'Hello Universe!';
}
}
$o = new TheWorldIsNotEnough();
//输出 Hello Universe!
$o->sayHello();
?>
- 冲突的解决
如果两个trait都插入了一个同名的方法,如果没有明确解决冲突将会产生一个致命错误。
当多个trait 在同一个类中引入,且存在方法名称重复时,则需要使用insteadof操作符来明确指定重名方法是属于那个trait的;
还可以使用as操作符将其中一个方法重命名。
如下代码:
<?php
trait A {
public function smallTalk() {
echo 'a';
}
public function bigTalk() {
echo 'A';
}
}
trait B {
public function smallTalk() {
echo 'b';
}
public function bigTalk() {
echo 'B';
}
}
class Talker {
//引入多个trait时,可以用逗号分割
use A, B {
//指定重名的方法分别属于那个trait
B::smallTalk insteadof A;
A::bigTalk insteadof B;
}
}
class Aliased_Talker {
use A, B {
//指定重名的方法分别属于那个trait
B::smallTalk insteadof A;
A::bigTalk insteadof B;
//用as 操作符重命名trait中的重复方法
B::bigTalk as talk;
}
}
?>
- 修改方法的访问控制
使用as操作符可以修改trait中方法的访问控制。
如下所示:
<?php
trait HelloWorld {
public function sayHello() {
echo 'Hello World!';
}
}
class MyClass1 {
//修改 sayHello 的访问控制
use HelloWorld { sayHello as protected; }
}
// 给方法一个改变了访问控制的别名
// 原版 sayHello 的访问控制则没有发生变化
class MyClass2 {
use HelloWorld { sayHello as private myPrivateHello; }
}
?>
对于在类中引入trait的其他操作访问,都与操作与同一个类中的方法、属性等相同。
注:trait 就相当于某一个class或者trait的一部分,在class与trait中引入某个trait时,会和当前引入的class或trait进行合并。