PHP后期静态绑定的(late static bindings)

理解PHP延迟静态绑定

static::中的static其实是运行时所在类的别名,并不是定义类时所在的那个类名。这个东西可以实现在父类中能够调用子类的方法和属性。

使用(static)关键字来表示这个别名,和静态方法,静态类没有半毛钱的关系,static::不仅支持静态类,还支持对象(动态类)。

转发调用:
所谓的“转发调用”(forwarding call)指的是通过以下几种方式进行的静态调用,self::,parent::,static::以及forward_static_call()。

非转发调用
那么非转发调用其实就是明确指定类名的静态调用(foo::bar())和非静态调用($foo->bar())

后期静态绑定原理
后期静态绑定工作原理是存储了在上一个“非转发调用”(non-forwarding call)的类名。

例子1,简单实用static

class A
{
public static function who()
{
echo CLASS;
}

public static function test()
{
	static::who();// 后期静态绑定从这里开始
}

}

class B extends A{
public static function who()
{
echo CLASS;
}
}

B::test();

以上程序会输出:
B

例子2,区分转发和非转发调用

class A
{
public static function foo()
{
static::who();
}

public static function who()
{
	echo __CLASS__;
}

}

class B extends A
{
public static function test()
{
A::foo();
parent::foo();
self::foo();
}

public static function who()
{
	echo __CLASS__;
}

}

class C extends B
{
public static function who()
{
echo CLASS;
}
}

C::test();

以上程序会输出:

A C C

在这里主要分析下例子2。

1、C::test(),这是一个非转发调用,因为::前面有类名C。
2、进入test()方法,有三个静态调用A::foo(),parent::foo(),self::foo(),对于这三个静态调用来说,它们的非转发调用类就是C。
3、现在执行A::foo(),这是一个非转发调用。A::foo()中的代码是static::who(),这是一个转发调用,对于这个转发调用来说他的非转发调用类就是不再
是C而是A(因为之前执行了A::foo())。因此执行的结果为A。
4、现在执行parent::foo(),这是一个转发调用,转发到哪里呢?就是它的上一个非转发调用的类,也就是C
(在步骤2中提到的)。在这里一定要注意虽然在这之前执行了A::foo(),但是parent::foo()的上一个非转发调用的类任然是类C,
因此执行的结果是C。
5、现在执行self::foo(),这个和parent::foo()一样都是转发调用,因此也输出C。

使用后期静态绑定的好处

后期静态绑定目前我看到家多的是用于对象实例化中,在实例化对象时,static会根据运行时调用的类来决定实例化对象,
而self则是分局所在位置的类来决定实例化对象。当我们只想实例化子类,并且不希望后续再对子类的使用中由于父类的变化对
子类产生了影响时,后期静态绑定就能发挥它的作用了。