trait 是类似于类的结构,它本身不能被实例化,但合一混合到类中。trait中定义的任何方法都可以被使用它的类所使用。trait可以改变类的结构,但无法更改其类型。我们可以将trait视为包含到类中的部分。
这里的例子主要作用是去除重复代码
//计算费率
trait priceUtilitiesTrait
{
private $taxRate = 10;
public function calculateTax(float $price) : float
{
return (($this->taxRate /100 ) * $price);
}
}
class ShopProduct
{
use priceUtilitiesTrait;
}
class UtilityService
{
use priceUtilitiesTrait;
}
实例化这两个类都可以调用到calculateTax方法
$shop = new ShopProduct();
print $shop->calculateTax(38.5)."\n";
$service = new UtilityService();
print $service->calculateTax(38.5)."\n";
从以上代码可以看出就是将ShopProduct类与UtilityService类公共的部分提练出来,在需要的类中use进来。
可以通过在use关键字后列举以逗号分隔的trait,在类中引用多个trait。
我们在定义饮用一个新的trait,IdentityTrait
trait IdentityTrait
{
public function generateId() : string
{
return uniqid();
}
}
通过use关键字引入IdentityTrait 和 priceUtilitiesTrait
class ShopProduct
{
use priceUtilitiesTrait, IdentityTrait;
}
那么ShopProduct类就同时拥有generateId() 方法 与calculateTax()方法了。
定义接口
interface IdentityObject
{
public function generateId() : string;
}
class ShopProduct implements IdentityObject
{
use IdentityTrait, priceUtilitiesTrait;
}
ShopProduct类实现了 IdentityObject 接口 同时引用了IdentityTrait,而引用的generateId() 【这里的generateId()指的是IdentityTrait 的】以满足了IdentityObject接口的需求。
我们在定义一个trait
trait TaxTools
{
public function calculateTax(float $price) : float
{
return 100;
}
}
然后使用ShopProduct类use TaxTools类与priceUtilitiesTrait类,然后用UtilityService的实例去调用calculateTax()方法
$utilityService = new UtilityService();
$utilityService->calculateTax(100);
原因: 引入的两个trait中都有calculateTax()方法,PHP不知道应该用哪一个。
同时如果使用ide的话,会有红色波浪线标红,我这里使用的是phpstorm(鼠标放在红线位置会给出错误信息)
看看世界上最好的语言是怎么处理这个问题的。
class UtilityService
{
use priceUtilitiesTrait,TaxTools{
TaxTools::calculateTax insteadof priceUtilitiesTrait;
}
}
就是在use语句中添加了一个花括号的语句块,然后由insteadof运算符链接组成,语句大概的意思就是用 谁的 什么方法 覆盖 谁的 同名方法
TaxTools::calculateTax insteadof priceUtilitiesTrait 的意思就是用TaxTools 的calculateTax()方法去代替priceUtilitiesTrait的calculateTax方法 ,
其实也就是当UtilityService实例调用calculateTax方法时,告给他应该调用谁的。
当然也可以理解为将 priceUtilitiesTrait的calculateTax方法 在UtilityService类中动态注释掉。
insteadof 是 覆盖另外一个trait的同名方法了,但是如果两个都要使用到呢,除了去手动修改方法名外,我们还可以动态的去改,也就是重写trait方法名
class UtilityService
{
use priceUtilitiesTrait,TaxTools{
TaxTools::calculateTax insteadof priceUtilitiesTrait;
priceUtilitiesTrait::calculateTax as basicTax;
}
}
一看as基本上就知道怎么回事了。
注意:当多个trait之间发生方法名冲突时,仅通过use代码给其中一个方法起别名时不够的,必须先使用insteadof操作符决定用那个方法覆盖那个方法,在使用as给覆盖的名字取一个新名字
as除了可以当作别名使用之外,还可以修改trait中方法的访问权限,例如,我们只想calculateTax方法在UtilityService内部使用,不想让外部的实现代码 使用调用,我们可以这样写:
class UtilityService
{
use priceUtilitiesTrait{
priceUtilitiesTrait::calculateTax as private;
}
}
当然也可以和别名一块使用:
class UtilityService
{
use priceUtilitiesTrait,TaxTools{
TaxTools::calculateTax insteadof priceUtilitiesTrait;
priceUtilitiesTrait::calculateTax as basicTax;
priceUtilitiesTrait::calculateTax as private;
}
}