111

我想更好地理解为什么在下面的场景中,类常量的继承方式与实例变量的方式有所不同。

<?php
class ParentClass {
    const TEST = "ONE";
    protected $test = "ONE";

    public function showTest(){
        echo self::TEST;
        echo $this->test;
    }
}

class ChildClass extends ParentClass {
    const TEST = "TWO";
    protected $test = "TWO";

    public function myTest(){
        echo self::TEST;
        echo $this->test;
    }
}

$child = new ChildClass();
$child->myTest();
$child->showTest();

输出:

TWO
TWO
ONE
TWO

在上面的代码中,ChildClass 没有 showTest() 方法,所以继承使用了 ParentClass 的 showTest() 方法。结果表明,由于该方法是在 ParentClass 上执行的,因此正在评估 TEST 常量的 ParentClass 版本,而因为它是通过继承在 ChildClass 上下文中评估的,所以正在评估 ChildClass 成员变量 $test。

我已经阅读了文档,但似乎没有看到任何关于这种细微差别的提及。任何人都可以为我解释一下吗?

4

2 回答 2

221

self::不是继承感知的,并且总是引用它正在执行的类。如果您使用的是 php5.3+,您可以尝试static::TEST继承static::感知。

不同之处在于static::使用“后期静态绑定”。在此处查找更多信息:

http://php.net/manual/en/language.oop5.late-static-bindings.php

这是我写的一个简单的测试脚本:

<?php

class One
{
    const TEST = "test1";

    function test() { echo static::TEST; }
}
class Two extends One
{
    const TEST = "test2";
}

$c = new Two();

$c->test();

输出

test2
于 2012-11-28T20:22:32.483 回答
19

在 PHP 中,self 指的是定义了被调用方法或属性的类。因此,在您的情况下,您正在调用selfin ChildClass,因此它使用该类中的变量。然后您使用selfin ParentClass,因此它将引用该类中的变量。

如果您仍然希望子类覆盖const父类的,则将父类中的以下代码调整为:

public function showTest(){
    echo static::TEST;
    echo $this->test;
}

注意static关键字。这是使用“后期静态绑定”。现在你的父类将调用你的子类的 const 。

于 2012-11-28T20:19:54.073 回答