3种策略巧妙化解PHP Trait成员属性冲突

说明

PHP语言本身可以用insteadof和as关键字解决多个trait同名成员方法冲突的问题,但是貌似没有直接解决同名成员属性冲突的方案。
虽然属性名冲突极少发生,但是不代表不会发生。

如果是自定义trait

  1. 可以复制旧trait文件到新trait,改新文件的成员属性名,引用新trait。
  2. 直接更改原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;

正确的解决方案

  1. 需要一个父类参与,相当于一个中间人为冲突双方做调解。
  2. 并在父类中引入任意一个trait,相当于告诉这个trait停止冲突。
  3. 子类继承父类并引入另一个trait,并重新声明属性并赋初始值,相当于告诉另一个trait也停止冲突,而且支持你。
  4. 此时子类继承了家业又化解了冲突。
  5. 需要注意: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,既不冲突,又能同时为我所用。

posted @ 2024-02-09 12:00  小松聊PHP进阶  阅读(135)  评论(0编辑  收藏  举报