前言
本文主要給大家介紹了關于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->aliases[$alias])) {
return class_alias($this->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->$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->instance('config', $config = new Repository($items));
是以 Config::get('app.name', 'dafault) 實際通路的是 Repository 實列的 get('app.name', 'default') 方法。