PHP trait 的理解和使用

trait

trait 是类似于类的结构,它本身不能被实例化,但合一混合到类中。trait中定义的任何方法都可以被使用它的类所使用。trait可以改变类的结构,但无法更改其类型。我们可以将trait视为包含到类中的部分。

1.定义和使用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进来。

2. 使用多个trait

可以通过在use关键字后列举以逗号分隔的trait,在类中引用多个trait。
我们在定义饮用一个新的trait,IdentityTrait

trait IdentityTrait
{
    public function generateId() : string
    {
        return uniqid();
    }
}

通过use关键字引入IdentityTrait 和 priceUtilitiesTrait

class ShopProduct
{
    use priceUtilitiesTrait, IdentityTrait;

}

那么ShopProduct类就同时拥有generateId() 方法 与calculateTax()方法了。

3.组合使用trait与接口

定义接口

interface IdentityObject
{
    public function generateId() : string;
}
class ShopProduct implements IdentityObject
{
    use IdentityTrait, priceUtilitiesTrait;
}

ShopProduct类实现了 IdentityObject 接口 同时引用了IdentityTrait,而引用的generateId() 【这里的generateId()指的是IdentityTrait 的】以满足了IdentityObject接口的需求。

4.通过insteadof管理方法名冲突使用别名重写trait的方法

我们在定义一个trait

trait TaxTools
{
    public function calculateTax(float $price) : float
    {
        return 100;
    }
}

然后使用ShopProduct类use TaxTools类与priceUtilitiesTrait类,然后用UtilityService的实例去调用calculateTax()方法

$utilityService = new UtilityService();
$utilityService->calculateTax(100);

php会报下方这个致命错误:
PHP trait 的理解和使用_第1张图片

原因: 引入的两个trait中都有calculateTax()方法,PHP不知道应该用哪一个

同时如果使用ide的话,会有红色波浪线标红,我这里使用的是phpstorm(鼠标放在红线位置会给出错误信息)
PHP trait 的理解和使用_第2张图片
看看世界上最好的语言是怎么处理这个问题的。

class UtilityService
{
    use priceUtilitiesTrait,TaxTools{
        TaxTools::calculateTax insteadof priceUtilitiesTrait;
    }
}

就是在use语句中添加了一个花括号的语句块,然后由insteadof运算符链接组成,语句大概的意思就是用 谁的 什么方法 覆盖 谁的 同名方法
TaxTools::calculateTax insteadof priceUtilitiesTrait 的意思就是用TaxToolscalculateTax()方法去代替priceUtilitiesTraitcalculateTax方法 ,
其实也就是当UtilityService实例调用calculateTax方法时,告给他应该调用谁的。
当然也可以理解为将 priceUtilitiesTraitcalculateTax方法 在UtilityService类中动态注释掉。

5.使用别名重写trait方法

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;
    }
}

你可能感兴趣的:(php,php)