php手册上关于后期静态绑定如下解释:
后期静态绑定工作原理是存储了在上一个"非转发调用"的类名。
这里我们先明确什么是"转发调用":
"转发调用"指的是通过以下几种方式进行的静态调用:self::, parent::, static::以及forward_static_call()。可以理解为没有指定类名的静态调用就是"转发调用"。
那么"非转发调用"就是指明确指定类名的静态调用和非静态调用。
该功能从语言内部角度考虑被命名为“后期静态绑定”。“后期绑定”的意思是说,static:: 不再被解析为定义当前方法所在的类,而是在实际运行时计算的。也可以称之为“静态绑定”,因为它可以用于(但不限于)静态方法的调用。
下面我们看看官网给的例子讲解:
class A {
public static function foo() {
static::who();
}
public static function who() {
echo __CLASS__."\n";
}
}
class B extends A {
public static function test() {
A::foo();
parent::foo();
self::foo();
}
public static function who() {
echo __CLASS__."\n";
}
}
class C extends B {
public static function who() {
echo __CLASS__."\n";
}
}
C::test();
分析这段代码:
C::test(),进入类B调用了A::foo(), parent::foo(), self::foo()
根据概念C::test()就是"非转发调用",类名C
通过A::foo()进入类A的foo方法,即'上一次非转发调用'的类变成A了,所以static::代表类A,A来调用who
parent::foo()进入类A的foo方法,但"上一次非转发调用"的类仍然为C,所以static::代表C,C来调用who
self::foo()进入类B的foo方法,"上一次非转发调用"的类还是C,所以static::代表C,C来调用who
所以得出:
A C C