Laravel中的Facades

先引用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做了一个代理工作,通过它来完成任务。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值