3种策略巧妙化解PHP Trait成员属性冲突
说明
PHP语言本身可以用insteadof和as关键字解决多个trait同名成员方法冲突的问题,但是貌似没有直接解决同名成员属性冲突的方案。
虽然属性名冲突极少发生,但是不代表不会发生。
如果是自定义trait
- 可以复制旧trait文件到新trait,改新文件的成员属性名,引用新trait。
- 直接更改原trait成员属性名,可能会影响项目。
如果就不动原trait,仅通过类怎么解决?(例如某些trait在vendor下,极少的情况下需要同时引用多个trait,他们的属性冲突了)
示例
如下:C类想用A Trait和B Trait的方法,但是属性名冲突,报错。
Fatal error: A and B define the same property ($prop) in the composition of C. However, the definition differs and is considered incompatible. Class was composed。
trait A {
public $prop = 'trait_a';
public function speakEnglish() {
echo 'English';
}
}
trait B {
public $prop = 'trait_b';
public function speakChinese() {
echo '中文';
}
}
class C {
use A,B;
}
$c = new C();
echo $c->prop;
错误的解决方案
PHP语言本身可以用insteadof和as关键字解决多个trait同名成员方法冲突的问题,但是这无法修饰成员属性。
报错:Fatal error: A precedence rule was defined for A::prop but this method does not exist.
class C {
use A,B {
A::prop insteadof B;
B::prop as B_prop;
}
}
$c = new C();
echo $c->prop;
正确的解决方案
- 需要一个父类参与,相当于一个中间人为冲突双方做调解。
- 并在父类中引入任意一个trait,相当于告诉这个trait停止冲突。
- 子类继承父类并引入另一个trait,并重新声明属性并赋初始值,相当于告诉另一个trait也停止冲突,而且支持你。
- 此时子类继承了家业又化解了冲突。
- 需要注意:C类中的
public $prop = 'trait_b'
不能少,且必须等于B trait中的值,否则会报致命错误。
trait A {
public $prop = 'trait_a';
public function speakEnglish() {
echo 'English';
}
}
trait B {
public $prop = 'trait_b';
public function speakChinese() {
echo '中文';
}
}
class P {
use A;
}
class C extends P {
use B;
public $prop = 'trait_b';
//构造方法为非必填项
public function __construct() {
$this->prop = 'new value';
}
}
$c = new C();
echo $c->prop;
$c->speakEnglish();
$c->speakChinese();
至此,使用父类可巧妙化解PHP Trait成员属性冲突的问题,让不能更改的两个trait,既不冲突,又能同时为我所用。