后期静态绑定

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

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

该功能从语言内部角度考虑被命名为“后期静态绑定”。“后期绑定”的意思是说,static:: 不再被解析为定义当前方法所在的类,而是在实际运行时计算的。也可以称之为“静态绑定”,因为它可以用于(但不限于)静态方法的调用。
self:: 的限制 使用 self:: 或者 __CLASS__ 对当前类的静态引用,取决于定义当前方法所在的类: Example #1 self:: 用法
<?php
class A {
public static function who() {
echo __CLASS__;
}
public static function test() {
self::who();
}
} class B extends A {
public static function who() {
echo __CLASS__;
}
} B::test();
?> 以上例程会输出: A 后期静态绑定的用法 后期静态绑定本想通过引入一个新的关键字表示运行时最初调用的类来绕过限制。简单地说,这个关键字能够让你在上述例子中调用 test() 时引用的类是 B 而不是 A。最终决定不引入新的关键字,而是使用已经预留的 static 关键字。 Example #2 static:: 简单用法
<?php
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 Note: 在非静态环境下,所调用的类即为该对象实例所属的类。由于 $this-> 会在同一作用范围内尝试调用私有方法,而 static:: 则可能给出不同结果。另一个区别是 static:: 只能用于静态属性。 Example #3 非静态环境下使用 static::
<?php
class A {
private function foo() {
echo "success!\n";
}
public function test() {
$this->foo();
static::foo();
}
} class B extends A {
/* foo() will be copied to B, hence its scope will still be A and
* the call be successful */
} class C extends A {
private function foo() {
/* original method is replaced; the scope of the new one is C */
}
} $b = new B();
$b->test();
$c = new C();
$c->test(); //fails
?> 以上例程会输出: success!
success!
success! Fatal error: Call to private method C::foo() from context 'A' in /tmp/test.php on line 9 Note: 后期静态绑定的解析会一直到取得一个完全解析了的静态调用为止。另一方面,如果静态调用使用 parent:: 或者 self:: 将转发调用信息。 Example #4 转发和非转发调用
<?php
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();
?> 以上例程会输出: A
C
C

最新文章

  1. DOTA 2 Match History WebAPI(翻译)
  2. 学习笔记:MySQL列属性
  3. 【读书笔记】.Net并行编程(三)---并行集合
  4. 利用SSIS的ForcedExecutionResult 属性 和CheckPoint调试Package
  5. Vue从零开始(一)
  6. CentOS配置SSH免密码登录后,仍提示输入密码
  7. 【css面试题】三个DIV要求水平对齐,左右两个DIV宽度固定为100px,中间那个DIV充满剩余的宽度(至少2种方法)
  8. 关于jquery 1.9以上多次点击checkbox无法选择的
  9. Win下安装Cygwin中的SSH服务
  10. day09 详解内存管理机制
  11. dbus-glib 和 GDBus 的区别
  12. FFMPEG 入门
  13. 11.1 正睿停课训练 Day14
  14. MySQL.ERROR 1133 (42000): Can&#39;t find any matching row in the user table
  15. 树莓派raspberry Pi2 介绍
  16. JAVA中定义常量的几种方式
  17. 记springboot+mybatis+freemarker+bootstrap的使用(2)
  18. blogCMS中出现的错误整理
  19. java内存管理和gc回收机制
  20. Shell--Bash shell的操作环境

热门文章

  1. 关于Cocos2d-x数据类型的使用
  2. Git:如何为项目设置自己的user name/email
  3. e1086. if/else语句
  4. 【c语言】推断一个数是奇偶数
  5. thinkphp隐藏模块实例
  6. MySQL数据库行去重复
  7. R基本介绍
  8. POSIX是什么?
  9. Extjs3.4--TabpanelDemo
  10. swift - UIDatePicker 的用法