天天看點

Laravel中Facade的加載過程與原理詳解

前言

本文主要給大家介紹了關于Laravel中Facade加載過程與原理的相關内容,分享出來供大家參考學習,下面話不多說了,來一起看看詳細的介紹吧。

簡介

Facades(讀音:/fəˈsäd/ )為應用程式的 服務容器 中可用的類提供了一個「靜态」接口。你不必 use 一大串的命名空間,也不用執行個體化對象,就能通路對象的具體方法。

use Config;

class Test

{

public function index()

{

return Config::get('app.name');

}

}

Facade 的啟動與注冊

Facade 的啟動引導是在 IlluminateFoundationBootstrapRegisterFacades 中注冊的。

public function bootstrap(Application $app)

{

Facade::clearResolvedInstances();

Facade::setFacadeApplication($app);

AliasLoader::getInstance(array_merge(

$app->make('config')->get('app.aliases', []),

$app->make(PackageManifest::class)->aliases()

))->register();

}

預設的别名配置是從 app 配置檔案下的 aliases 讀取的,PackageManifest 是 laravel 5.5 新增的 包自動發現 規則,這裡我們暫時不考慮 PackageManifest 包提供的别名。

其中,array_merge 傳回如下格式的數組:

"App" => "IlluminateSupportFacadesApp"

"Artisan" => "IlluminateSupportFacadesArtisan"

"Auth" => "IlluminateSupportFacadesAuth"

"Blade" => "IlluminateSupportFacadesBlade"

...

上面代碼将通過 AliasLoader 把所有的 facade 注冊進自動加載。其核心就是 php 的 spl_autoload_register。

/**

  • Prepend the load method to the auto-loader stack.

    *

  • @return void

    */

protected function register()

{

if (! $this->registered) {

spl_autoload_register([$this, 'load'], true, true);

$this->registered = true;

}

}

注冊完成後,後續所有 use 的類都将通過 load 函數來完成類的自動加載。

注意:這裡在定義 spl_autoload_register 時,最後面的參數傳的是 true。當該參數是 true 時,spl_autoload_register() 會添加函數到隊列之首,而不是隊列尾部。(優先通過該函數來完成自動加載)

也就是說,

<?php

use Config;

use AppUser;

class Test

{

public function index()

{

Config::get('app.name');

new User();

}

}

不管我們 use 的是具體存在的類(AppUser)還是别名 (Config),都将最先通過 load 函數來完成自動加載,當該函數傳回 false 時,再由其他自動加載函數來完成自動加載(如 composer psr-4)。

在 AliasLoader 的 load 方法中,主要是用了 class_alias 函數來實作的别名自動加載。

public function load($alias)

{

if (isset($this-&gt;aliases[$alias])) {

return class_alias($this-&gt;aliases[$alias], $alias);

}

}

關于 class_alias 這裡帖一個官方的列子:

class foo { }

class_alias('foo', 'bar');

$/【盡量使用一鍵安裝腳本,要麼自己做,要麼網上下載下傳或使用我部落格的,把時間用在更多的地方,少做重複勞動的事情】/a = new foo;

$b = new bar;

// the objects are the same

var_dump($a == $b, $a === $b); //true

var_dump($a instanceof $b); //false

// the classes are the same

var_dump($a instanceof foo); //true

var_dump($a instanceof bar); //true

var_dump($b instanceof foo); //true

var_dump($b instanceof bar); //true

Facade 的加載

當我們在使用 Facade 時,如:

<?php

use Config;

class Test

{

public function index()

{

Config::get('app.name');

}

}

實際上加載的是 IlluminateSupportFacadesConfig 類(因為我們已經注冊了 class_alias),相當于:

<?php

use IlluminateSupportFacadesConfig;

class Test

{

public function index()

{

Config::get('app.name');

}

}

而所有的 Facade 都繼承自 IlluminateSupportFacadesFacade 類,在該基類中定義了一個 __callStatic 方法,已至于我們能夠輕松地使用 Facade(不用實列化)。

<?php

public static function __callStatic($method, $args)

{

$instance = static::getFacadeRoot();

if (! $instance) {

throw new RuntimeException('A facade root has not been set.');

}

return $instance-&gt;$method(...$args);

}

getFacadeRoot 方法用于擷取别名類的具體實列,我們知道,所有的 Facade 類都需要定義一個 getFacadeAccessor 方法。該方法可能的傳回值有:

String 類型的字元串(如 config, db)

String 類型的類字元串 (如 AppServiceSomeService)

Object 具體的實列化對象

Closure 閉包

如 Config Facade 的 getFacadeAccessor 方法如下:

protected static function getFacadeAccessor()

{

return 'config';

}

getFacadeRoot 方法将根據 getFacadeAccessor() 的傳回值,從容器從取出對應的實列對象。

public static function getFacadeRoot()

{

$name = static::getFacadeAccessor();

if (is_object($name)) {

return $name;

}

if (isset(static::$resolvedInstance[$name])) {

return static::$resolvedInstance[$name];

}

return static::$resolvedInstance[$name] = static::$app[$name];

}

由于 APP 容器中已經注冊過 config 的實列

<?php

//IlluminateFoundationBootstrap/LoadConfiguration

$app-&gt;instance('config', $config = new Repository($items));

是以 Config::get('app.name', 'dafault) 實際通路的是 Repository 實列的 get('app.name', 'default') 方法。