通过引用赋值静态属性在PHP5和PHP7之间的行为不同。

问题描述 投票:3回答:2

从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 php-7 late-static-binding
2个回答
4
投票

这在 PHP 7.2 => 7.3 中被修改了。

静态属性不再用引用分配来分隔。

在PHP中,静态属性在继承类之间是共享的,除非静态属性在子类中被显式地覆盖了。这个漏洞已经被修复。

https:/www.php.netmanualenmigration73.incompatible.php


2
投票

我想这是你的本地环境出了问题。

你可以用以下方法检查你的代码 php-sandbox.

我检查了一下,对于 php5php7 结果是一样的。

© www.soinside.com 2019 - 2024. All rights reserved.