前言
众所周知,一直以来PHP和很多语言一样是单继承的语言,但是常常在编码过程中,我们需要在当前类中使用两个或两个以上的其他类的方法,这种情况下继承就不能实现,而往往采用new方式实例化很多要用到的类,这样就会很影响代码的结构和开发规范。于是Trait类诞生了,它是一种代码复用的语法,能够实现一个类中引用多个其他类的方法。
一、概念
Traits 是一种为类似 PHP 的单继承语言而准备的代码复用机制。Trait 为了减少单继承语言的限制,使开发人员能够自由地在不同层次结构内独立的类中复用方法集。Traits 和类组合的语义是定义了一种方式来减少复杂性,避免传统多继承和混入类(Mixin)相关的典型问题。
Trait和Class相似,但仅仅旨在用细粒度和一致的方式来组合功能。无法通过trait自身来实例化。它为传统继承增加了水平特性的组合;也就是说,应用的几个Class之间不需要继承。
二、Trait类的使用
简单地讲,Trait就是一种不同于继承的语法,定义一个trait类,在其他类中使用它则是采用use关键字,有点类似于命名空间的用法,但是含义不同。use关键字在一个类中引入Trait类后,相当于require或include了一段代码进来,不同之处在于use的Trait类与当前类是可以看做同一个类的,即当前类可以用$this关键字调用Trait类的方法。
三、Trait类的访问控制
我们知道,继承的方式,如果基类是private修饰控制的,则子类是无法调用的。但是Trait不一样,因为它类似于Require到当前类中了,所以不管是public、protected或private都是可以直接使用的。示例如下:
四、Trait类的优先级控制
Trait类与当前使用类、继承的基类之间的调用优先级顺序如下:
当前使用类>Trait类>继承的基类
当存在同名方法时,会根据优先级覆盖掉同名的类。具体示例如下:
1、Trait类覆盖基类
2、当前类覆盖Trait类
五、多个Trait类的冲突控制
在PHP中,如果当前类use了两个Trait类,同时两个trait类都存在一个同名的方法,此时如果没有明确解决冲突将会产生一个致命错误。
对于这种情况,PHP官方给出了两个解决方案:
1、insteadof关键字:通过该关键字指定方法名冲突时该使用哪个Trait类的方法,即:
如果C类use了A、B两个Trait类,且A、B两个类都存在a、b方法,则在C类use A、B类时使用insteadof声明冲突的解决方法即可:
use A, B {
B::a insteadof A; //a方法冲突时使用B类的a方法而不使用A类的a方法
A::b insteadof B; //b方法冲突时使用A类的b方法而不使用B类的b方法
}
2、as关键字:通过as关键字将同名方法指定为一个别名,且仅作用于当前类中。示例如下:
use A, B {
B::a as c; //声明B类的a方法为c,作用于该类
A::b as d; //声明A类的b方法为d,作用于该类
}
六、与继承、直接实例化的区别
对于当前一个类需要用到另一个或多个类的方法的情况,我们一般会想到的方式有继承、直接实例化另外一个或多个类等等的方法,下面来对比一下这些方法和Trait类的区别:
1、继承方式:对于继承,可以完美地复用另一个类的一些方法,但是对于需要复用多个类的方法时,PHP是不支持多继承的,而且只能访问public和protected方法;
2、与直接实例化的区别:我们也可以在当前类中直接实例化要用到的A类与B类,但是这种方法在控制访问范围反面,只允许访问A、B类中public的方法;
3、使用Trait类则完全将A、B两个类的方法导入到当前类中,可以视为当前类的一部分,唯一区别是可能存在于当前类同名的方法,此时由优先级顺序来控制。
补充:PHP多继承示例
class Base{
public function sayHello(){
echo "hello ";
}
}
trait SayWorld{
public function sayHello(){
parent::sayHello();
echo "world".PHP_EOL;
}
}
trait SayWorld2{
public function sayHello2(){
echo "PHP".PHP_EOL;
}
}
class MyHelloWorld extends Base{
use SayWorld,SayWorld2;
}
$s = new MyHelloWorld();
$s->sayHello();
$s->sayHello2();
输出结果:
hello world
PHP
上面就是些 Trait 比较基本的使用了。这里总结下注意的几点:
1.Trait 会覆盖调用类继承的父类方法,但也会被当前类所覆盖
2.Trait 无法如 Class 一样使用 new 实例化
3.单个 Trait 可由多个 Trait 组成
4.在单个 Class 中,可以使用多个 Trait
5.Trait 支持修饰词(modifiers),例如 final、static、abstract
6.我们能使用 insteadof 以及 as 操作符解决 Trait 之间的冲突
7.Trait中不区分修饰符,即可以操作Trait中的public protected private级别的属性和方法,这个extends继承有所不同