从PHP5升级到PHP7后,下面的代码产生了不同的结果。
abstract class TheParent {
public static $prop;
public function __construct( $val ) {
static::set_prop($val);
}
public static function set_prop( $val ) {
$ref_reset = $val;
static::$prop =& $ref_reset;
}
}
class Child extends TheParent {
// public static $prop; //<-- not declared on purpose
}
class Child2 extends TheParent {
// public static $prop; //<-- not declared on purpose
}
$c = new Child('do');
$c2 = new Child2('re');
echo 'Child: ' . Child::$prop . '<br>';
echo 'Child2: ' . Child2::$prop . '<br>';
echo 'TheParent: ' . TheParent::$prop;
在PHP5中。
Child: do
Child2: re
TheParent:
在PHP7中。
Child: re
Child2: re
TheParent: re
我想要的输出是PHP5的输出,因为我希望能够在所有扩展基类(父类)的类中引用一个单一的属性名,但我不想在每一个子类中重新声明该属性或设置它的方法(主要是为了避免在几十个类中添加相同的属性方法的维护开销)。
看来PHP5中的神奇之处在于通过引用赋值(在SO.NET上猎奇了很久)。有益 回答者 已经提到,通过引用设置 "打破""引用集",允许每个子类中的属性持有单独的值)。) 我发现这在PHP5中是一个非常优雅的解决方案。
有没有办法在PHP7中用相同或类似的代码实现同样的结果?
看起来这是PHP7.2和7.3之间的一个突破性变化的结果,可能没有一个类似的优雅的双行替代方案。在重构了我的代码之后,我发现这个略显啰嗦的工作方法很有效(并且满足了我的主要目标,即不必在子类中重新声明属性)。
abstract class TheParent {
public static $props = [];
public function __construct( $val ) {
static::set_prop($val);
}
public static function set_prop( $val ) {
self::$props[static::class] = $val;
}
public static function get_prop() {
if( isset(self::$props[static::class]) )
return self::$props[static::class];
}
}
class Child extends TheParent {
// public static $prop; //<-- not declared on purpose
}
class Child2 extends TheParent {
// public static $prop; //<-- not declared on purpose
}
$c = new Child('do');
$c2 = new Child2('re');
echo 'Child: ' . Child::get_prop(). '<br>'; // 'do' in PHP 7.3
echo 'Child2: ' . Child2::get_prop() . '<br>'; // 're' in PHP 7.3
echo 'TheParent: ' . TheParent::get_prop(); // '' in PHP 7.3
这在 PHP 7.2 => 7.3 中被修改了。
静态属性不再用引用分配来分隔。
在PHP中,静态属性在继承类之间是共享的,除非静态属性在子类中被显式地覆盖了。这个漏洞已经被修复。