PHP类与对象后期静态绑定操作实例详解

本文实例讲述了PHP类与对象后期静态绑定操作,分享给大家供大家参考,具体如下:

做项目是后期静态绑定非常有用,比如service层单例模式,使用后期静态绑定就非常好实现。

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

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

该功能从语言内部角度考虑被命名为“后期静态绑定”。“后期绑定”的意思是说,static:: 不再被解析为定义当前方法所在的类,而是在实际运行时计算的。也可以称之为“静态绑定”,因为它可以用于(但不限于)静态方法的调用。

  1. <?php
  2. class A {
  3. public static function who() {
  4. echo __CLASS__;
  5. }
  6. public static function test() {
  7. static::who(); // 后期静态绑定从这里开始
  8. }
  9. }
  10. class B extends A {
  11. public static function who() {
  12. echo __CLASS__;
  13. }
  14. }
  15. B::test();
  16. ?>

以上例程会输出:

B

Note:

在非静态环境下,所调用的类即为该对象实例所属的类。由于 $this-> 会在同一作用范围内尝试调用私有方法,而 static:: 则可能给出不同结果。另一个区别是 static:: 只能用于静态属性。

Example #3 非静态环境下使用 static::

  1. <?php
  2. class A {
  3. private function foo() {
  4. echo "success!\n";
  5. }
  6. public function test() {
  7. $this->foo();
  8. static::foo();
  9. }
  10. }
  11. class B extends A {
  12. /* foo() will be copied to B, hence its scope will still be A and
  13. * the call be successful */
  14. }
  15. class C extends A {
  16. private function foo() {
  17. /* original method is replaced; the scope of the new one is C */
  18. }
  19. }
  20. $b = new B();
  21. $b->test();
  22. $c = new C();
  23. $c->test(); //fails
  24. ?>

以上例程会输出:

success!

success!

success!

Fatal error: Call to private method C::foo() from context ‘A' in /tmp/test.php on line 7

Note:

后期静态绑定的解析会一直到取得一个完全解析了的静态调用为止,另一方面,如果静态调用使用 parent:: 或者 self:: 将转发调用信息。

Example #4 转发和非转发调用

  1. <?php
  2. class A {
  3. public static function foo() {
  4. static::who();
  5. }
  6. public static function who() {
  7. echo __CLASS__."\n";
  8. }
  9. }
  10. class B extends A {
  11. public static function test() {
  12. A::foo();
  13. parent::foo();
  14. self::foo();
  15. }
  16. public static function who() {
  17. echo __CLASS__."\n";
  18. }
  19. }
  20. class C extends B {
  21. public static function who() {
  22. echo __CLASS__."\n";
  23. }
  24. }
  25. C::test();
  26. ?>

以上例程会输出:

A

C

C