什么叫前期动态绑定呢?其实咱们在之前的文章PHP中的static中曾经说过这个货色了。明天咱们还是再次深刻的了解一下这个概念。
首先,咱们通过一段代码来引入前期动态绑定这一概念:
【腾讯云】云产品限时秒杀,爆款1核2G云服务器,首年99元
class A
{
public static function who()
{
echo __CLASS__, PHP_EOL;
}
public static function test()
{
self::who();
}
}
class B extends A
{
public static function who()
{
echo __CLASS__, PHP_EOL;
}
}
B::test(); // A
在这段代码中,咱们应用了self关键字,当应用B类调用test()静态方法时,self指向的是A类的who()办法,因而,输入的是A。别冲动,这是一般的动态绑定。self关键字调用的内容取决于它定义时所在的类。也就是说不管怎么继承,用哪个子类来调用test()办法,self关键字都会调用的是A类的who()办法。
而前期动态绑定呢?其实就有点像实例化的类对象,每个实例化的对象,调用的都是本身,而不是父类的属性办法。一般的动态调用可不是这样,然而事实中咱们又有这样的需要,就像实例化对象的调用形式一样来调用动态属性办法,这时,咱们就能够应用static关键字来实现前期动态绑定。
class C
{
public static function who()
{
echo __CLASS__, PHP_EOL;
}
public static function test()
{
static::who();
}
}
class D extends C
{
public static function who()
{
echo __CLASS__, PHP_EOL;
}
}
D::test(); // D
当应用static关键字后,这里D类调用的test()办法外部调用的who()就是D类本人了。
官网文档中的定义如下:
当进行静态方法调用时,该类名即为明确指定的那个(通常在 :: 运算符左侧局部);当进行非静态方法调用时,即为该对象所属的类。
该性能从语言外部角度思考被命名为“前期动态绑定”。“前期绑定”的意思是说,static:: 不再被解析为定义以后办法所在的类,而是在理论运行时计算的。也能够称之为“动态绑定”,因为它能够用于(但不限于)静态方法的调用。
除了self和static关键字外,咱们还有一个parent关键字,这个关键字的意义就很显著了,调用父类的动态内容。咱们同时用三个关键字一起来进行测试:
class E
{
public static function who()
{
echo __CLASS__, PHP_EOL;
}
public static function test()
{
self::who();
static::who();
}
}
class F extends E
{
public static function who()
{
echo __CLASS__, PHP_EOL;
}
}
class G extends F
{
public static function who()
{
parent::who();
echo __CLASS__, PHP_EOL;
}
}
G::test();
// E
// F
// G
最初,咱们再来看两个PHP的办法,一个是get_called_class()办法,用来获取以后调用的是哪个类。在静态方法中能够依据调用形式判断以后类是哪个类来进行其余的业务逻辑操作。另一个是forward_static_call()办法,用于静态方法的调用。
class H
{
public static function who()
{
echo __CLASS__ . ':' . join(',', func_get_args()), PHP_EOL;
}
public static function test()
{
echo get_called_class(), PHP_EOL;
forward_static_call('who', 'a', 'b'); // xxx:a,b
forward_static_call(['I', 'who'], 'c', 'd'); // I:c,d
forward_static_call_array(['H', 'who'], ['e', 'f']); // H:e,f
}
}
class I extends H
{
public static function who()
{
echo __CLASS__ . ':' . join(',', func_get_args()), PHP_EOL;
}
}
function who()
{
echo 'xxx:' . join(',', func_get_args()), PHP_EOL;
}
H::test(); // H
// xxx:a,b
// I:c,d
// H:e,f
I::test(); // I
// xxx:a,b
// I:c,d
// H:e,f
留神,如果forward_static_call()不指定类名的话,将调用全局的办法。forward_static_call_array()则是将参数应用数组进行传递。
测试代码:
https://github.com/zhangyue0503/dev-blog/blob/master/php/202001/source/%E5%90%8E%E6%9C%9F%E9%9D%99%E6%80%81%E7%BB%91%E5%AE%9A%E5%9C%A8PHP%E4%B8%AD%E7%9A%84%E4%BD%BF%E7%94%A8.php
参考文档:
https://www.php.net/manual/zh/language.oop5.late-static-bindings.php
https://www.php.net/manual/zh/function.get-called-class.php
https://www.php.net/manual/zh/function.forward-static-call.php
各自媒体平台均可搜寻【硬核项目经理】