php后期静态绑定

自 PHP 5.3.0 起,PHP 增加了一个叫做后期静态绑定的功能,用于在继承范围内引用静态调用的类。

准确说,后期静态绑定工作原理是存储了在上一个“非转发调用”(non-forwarding call)的类名。当进行静态方法调用时,该类名即为明确指定的那个(通常在 :: 运算符左侧部分);当进行非静态方法调用时,即为该对象所属的类。所谓的“转发调用”(forwarding call)指的是通过以下几种方式进行的静态调

用:self::,parent::,static:: 以及 forward_static_call()。可用 get_called_class() 函数来得到被调用的 方法所在的类名,static:: 则指出了其范围。

该功能从语言内部角度考虑被命名为“后期静态绑定”。“后期绑定”的意思是说,static:: 不再被解析为定义当前方法所在的类,而是在实际运行时计算的。也可以称之为“静态绑定”,因为它可以用于(但不 限于)静态方法的调用。

代码如下:

class A
{
    public static $name = "good\n";
    public static function test()
    {
        echo 'A'."\n";
    }
}
class B extends A
{
    public static $name = "hello\n";
    public static function test()
    {
        echo 'B'."\n";
    }

    public static function say()
    {
        self::test();
        static::test();
        parent::test();
        echo self::$name;
        echo static::$name;
    }
}
class C extends B
{
    public static $name = "I'm C\n";
    public static function test()
    {
        echo "C\n";
    }
}

运行 C::say();

运行结果:

PHP static

通过以上例子我们发现,在B类的say()方法中:

  • 1.self指向本类(也就是B类)的test()方法;
  • 2.由于C类继承B类,而且是通过C类发起的say()方法调用,通过继承关系可以看到,C类中的test()方法覆写了它的父类(B类)的test()方法,而此时的B类的say()方法中调用的static::test()结果输出了"C",因此不难看出static使用了"后一个"静态方法;
  • 3.self::$name和static::$name的情况同上述1,2中的方法调用情况一致。
    最后再看一遍文档上的说明:

后期绑定”的意思是说,static:: 不再被解析为定义当前方法所在的类,而是在实际运行时计算的。代码结论中的 "后一个" 可以理解为文档说明中的 "实际运行时的"。