最近在看laravel的源码,开始的时候static关键字的用法让我读完一头雾水,
static不应该是用来定义静态变量方法和属性的么?
在方法中
static::setInstance($this);
是什么?
查了手册才明白,这是static关键字的另一种用法,“后期静态绑定”,也可以称为“静态绑定”
http://php.net/manual/zh/language.oop5.late-static-bindings.php
官方的介绍:
该功能从语言内部角度考虑被命名为“后期静态绑定”。“后期绑定”的意思是说,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
在这里static相当于self,parent一样的关键字,用来调用方法,不过与self不同,static调用的类并不一定是静态属性和静态方法,这点上看更接近于parent这个关键字的用法。
but,在下面的例子了还看到这样一种用法
<?php class Model { public static function find() { echo static::$name; } } class Product extends Model { protected static $name = ‘Product‘; } Product::find(); ?>
在父类里并未定义$name这个属性,这时候在子类里必须
static $name
不过一般程序应该不会出现这种父类没定义属性而直接使用的情况吧。
-----------------------------------------------------------------------------------------------------------
另外,测试例子的时候想到这么一个情况
class A { public static function who() { echo __CLASS__; } public function test() { static::who(); } } class B extends A { public static function who() { echo __CLASS__; } } (new B())->test();
非静态方法可以调用静态方法,反之则不行
laravel设计模式中大量使用了这种静态绑定的方法,尤其是核心功能Facades。
php面向对象编程还有很多东西需要了解。。。