什么是 trait?
看看 PHP 官网的介绍。
自 PHP 5.4.0 起,PHP 实现了一种代码复用的方法,称为 trait。
Trait 是为类似 PHP 的单继承语言而准备的一种代码复用机制。Trait 为了减少单继承语言的限制,使开发人员能够自由地在不同层次结构内独立的类中复用 method。Trait 和 Class 组合的语义定义了一种减少复杂性的方式,避免传统多继承和 Mixin 类相关典型问题。
Trait 和 Class 相似,但仅仅旨在用细粒度和一致的方式来组合功能。 无法通过 trait 自身来实例化。它为传统继承增加了水平特性的组合;也就是说,应用的几个 Class 之间不需要继承。
例子1
手机和汽车都有 GPS 功能,GPS 是用来定位的功能的,因此功能应该是统一的。手机 和 汽车 除了 GPS 功能外,基本没有什么相同之处,因此不能使用继承。而接口的话,我认为不同的类去实现接口时,接口的实现可能是不同的,但是 GPS 的功能就是用来定位的。因此使用 trait ,而不使用 class 和 interface,这是我的理解,不知道是否正确。
gps.php 的定义:
<?php
trait Gps {
public function gps() {
echo 'i can gps';
}
}
使用 trait 定义了一个 GPS 的 trait 用于复用,它的关键字是 trait 。然后在 car.php 和 mobile.php 中进行引用。
car.php 的定义:
class Car {
use gps;
public function move() {
echo 'i can move';
}
}
mobile.php 的定义:
<?php
class Mobile {
use gps;
public function tel() {
echo 'i can tel';
}
}
在 car.php 和 mobile.php 中,使用 use 关键字引入了 gps 的 trait ,这样在 car 和 mobile 中就可以调用 gps() 这个方法了。
test.php 进行测试:
<?php
require_once('gps.php');
require_once('car.php');
require_once('mobile.php');
$car = new Car();
$mobile = new Mobile();
$car->gps();
echo "\n";
$mobile->gps();
输出结果如下:
1 i can gps
2 i can gps
例子2
在 car 中引入了另外一个国产的 gps 。
gpschina.php 定义如下:
<?php
trait GpsChina {
public function gps() {
echo 'i can chinae gps';
}
}
在 car 中引入,修改 car.php 的定义如下:
<?php
class Car {
use gps, gpschina;
public function move() {
echo 'i can move';
}
}
再次调用 test.php 进行测试,这时会报错,报错如下:
1 Fatal error: Trait method gps has not been applied, because there are collisions with other trait methods on Car in Car.php on line 4
因为在引入的 trait 中 gps 和 gpschina 各有一个 gps ,而直接使用 $car->gps() 时无法确定到底使用的是 gps 的 gps() 方法,还是使用的 gpschina 的 gps() 方法,因此报错了。这样的话,我们需要确定一个。修改 car.php 文件。
<?php
class Car {
use gps, gpschina {
GpsChina::gps insteadof Gps;
}
public function move() {
echo 'i can move';
}
}
这样就使用 GpsChina::gps 的方法 替换掉了 Gps 的方法了,在调用 test.php 进行查看。
1 i can chinae gps
2 i can gps
这样,对于 $car->gps() 后就调用了 gpschina 中的 gps() 方法了。
例子3
如果在 Car 类中本身有一个 gps() 方法呢?修改 Car 类。
<?php
class Car {
use gps, gpschina {
GpsChina::gps insteadof Gps;
}
public function gps() {
echo 'car::gps';
}
public function move() {
echo 'i can move';
}
}
调用 test.php 查看结果:
1 car::gps
2 i can gps
可以看出,调用了 Car 类本身的 gps() 方法。
如果在一个类中,继承自父类的方法、use 引入 trait 的方法 和 类自身的方法同名的话,优先调用 自身类的 方法,如果没有 自身类的方法 则调用 use 引入 trait 的方法,如果前两个都没有,那么就调用继承自父类的方法。