先引用Laravel
文档中的解释:
Facades 为应用程序的 服务容器 中可用的类提供了一个「静态」接口。Laravel 本身附带许多的 facades,甚至你可能在不知情的状况下已经在使用他们!Laravel 「facades」作为在服务容器内基类的「静态代理」,拥有简洁、易表达的语法优点,同时维持着比传统静态方法更高的可测试性和灵活性。
继承Facades
的类,可以像使用静态方法一样调用一个方法,但是这个方法并不属于该类。
有点拗口,举个例子:
Laravel
中可以通过DB::select(..)
从表中查询数据,我们可以查看一下DB
:
class DB extends Facade
{
protected static function getFacadeAccessor()
{
return 'db';
}
}
代码很少,而且不存在select()
方法,那为什么我们能够使用select()
方法呢??
能够这样做要归功于 __callStatic()
魔术方法,当调用类的静态方法时,如果该方法不存在,会调用__callStatic()
方法,现在查看父类Facade
的__callStatic()
:
public static function __callStatic($method, $args)
{
$instance = static::getFacadeRoot();
if (! $instance) {
throw new RuntimeException('A facade root has not been set.');
}
switch (count($args)) {
case 0:
return $instance->$method();
case 1:
return $instance->$method($args[0]);
case 2:
return $instance->$method($args[0], $args[1]);
case 3:
return $instance->$method($args[0], $args[1], $args[2]);
case 4:
return $instance->$method($args[0], $args[1], $args[2], $args[3]);
default:
return call_user_func_array([$instance, $method], $args);
}
}
它的参数$method
表示调用的不存在的静态方法,$args
表示参数。
这里先看一下static::getFacadeRoot()
:
public static function getFacadeRoot()
{
return static::resolveFacadeInstance(static::getFacadeAccessor());
}
可以看到这里调用了getFacadeAccessor()
方法,DB
类中我们重写了该方法,返回了’db’,这里将’db’作为参数传给resolveFacadeInstance()
, 很有必要看一下这个方法:
protected static function resolveFacadeInstance($name)
{
if (is_object($name)) {
return $name;
}
if (isset(static::$resolvedInstance[$name])) {
return static::$resolvedInstance[$name];
}
return static::$resolvedInstance[$name] = static::$app[$name];
}
可以看到在这里我们会通过容器获取该注册名对应的实例,也就是这里我们获取了’db’对应的实例。
现在回到__callStatic()
方法中,$instance = static::getFacadeRoot()
得到了我们需要的对象实例,然后在调用该实例的$method
方法。
也就是说Facade
做了一个代理工作,通过它来完成任务。