自 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();
运行结果:
通过以上例子我们发现,在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:: 不再被解析为定义当前方法所在的类,而是在实际运行时计算的。代码结论中的 "后一个" 可以理解为文档说明中的 "实际运行时的"。