国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

Laravel框架門面Facade源碼分析

wanghui / 2204人閱讀

摘要:容器主要的作用就是生產(chǎn)各種零件,就是提供各個(gè)服務(wù)。的原理我們以為例,來講解一下門面的原理與實(shí)現(xiàn)。當(dāng)運(yùn)行時(shí),發(fā)現(xiàn)門面沒有靜態(tài)函數(shù),就會調(diào)用這個(gè)魔術(shù)函數(shù)。我們看到這個(gè)魔術(shù)函數(shù)做了兩件事獲得對象實(shí)例,利用對象調(diào)用函數(shù)。

前言

在開始之前,歡迎關(guān)注我自己的博客:www.leoyang90.cn
這篇文章我們開始講 laravel 框架中的門面 Facade,什么是門面呢?官方文檔:

Facades(讀音:/f??s?d/ )為應(yīng)用程序的服務(wù)容器中可用的類提供了一個(gè)「靜態(tài)」接口。Laravel 自帶了很多 facades ,幾乎可以用來訪問到 Laravel 中所有的服務(wù)。Laravel facades 實(shí)際上是服務(wù)容器中那些底層類的「靜態(tài)代理」,相比于傳統(tǒng)的靜態(tài)方法, facades 在提供了簡潔且豐富的語法同時(shí),還帶來了更好的可測試性和擴(kuò)展性。

什么意思呢?首先,我們要知道 laravel 框架的核心就是個(gè) Ioc 容器即 服務(wù)容器,功能類似于一個(gè)工廠模式,是個(gè)高級版的工廠。laravel 的其他功能例如路由、緩存、日志、數(shù)據(jù)庫其實(shí)都是類似于插件或者零件一樣,叫做 服務(wù)。Ioc 容器主要的作用就是生產(chǎn)各種零件,就是提供各個(gè)服務(wù)。在 laravel 中,如果我們想要用某個(gè)服務(wù),該怎么辦呢?最簡單的辦法就是調(diào)用服務(wù)容器的 make 函數(shù),或者利用依賴注入,或者就是今天要講的門面 Facade。門面相對于其他方法來說,最大的特點(diǎn)就是簡潔,例如我們經(jīng)常使用的 Router,如果利用服務(wù)容器的 make:

App::make("router")->get("/", function () {
  return view("welcome");
});

如果利用門面:

Route::get("/", function () {
  return view("welcome");
});

可以看出代碼更加簡潔。其實(shí),下面我們就會介紹門面最后調(diào)用的函數(shù)也是服務(wù)容器的 make 函數(shù)。

Facade 的原理

我們以 Route 為例,來講解一下門面 Facade 的原理與實(shí)現(xiàn)。我們先來看 Route 的門面類:

class Route extends Facade
{
    protected static function getFacadeAccessor()
    {
        return "router";
    }
}

很簡單吧?其實(shí)每個(gè)門面類也就是重定義一下 getFacadeAccessor 函數(shù)就行了,這個(gè)函數(shù)返回服務(wù)的唯一名稱:router。需要注意的是要確保這個(gè)名稱可以用服務(wù)容器的 make 函數(shù)創(chuàng)建成功(App::make("router")),原因我們馬上就會講到。
那么當(dāng)我們寫出 Route::get() 這樣的語句時(shí),到底發(fā)生了什么呢?奧秘就在基類 Facade中。

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);
}

當(dāng)運(yùn)行 Route::get() 時(shí),發(fā)現(xiàn)門面 Route 沒有靜態(tài) get() 函數(shù),PHP 就會調(diào)用這個(gè)魔術(shù)函數(shù) __callStatic。我們看到這個(gè)魔術(shù)函數(shù)做了兩件事:獲得對象實(shí)例,利用對象調(diào)用 get() 函數(shù)。首先先看看如何獲得對象實(shí)例的:

public static function getFacadeRoot()
{
    return static::resolveFacadeInstance(static::getFacadeAccessor());
}

protected static function getFacadeAccessor()
{
    throw new RuntimeException("Facade does not implement getFacadeAccessor method.");
}

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];
}

我們看到基類 getFacadeRoot() 調(diào)用了 getFacadeAccessor(),也就是我們的服務(wù)重載的函數(shù),如果調(diào)用了基類的 getFacadeAccessor,就會拋出異常。在我們的例子里 getFacadeAccessor() 返回了 “router”,接下來 getFacadeRoot() 又調(diào)用了 resolveFacadeInstance()。在這個(gè)函數(shù)里重點(diǎn)就是

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

我們看到,在這里利用了 app 也就是服務(wù)容器創(chuàng)建了 “router”,創(chuàng)建成功后放入 resolvedInstance作為緩存,以便以后快速加載。
好了,F(xiàn)acade 的原理到這里就講完了,但是到這里我們有個(gè)疑惑,為什么代碼中寫 Route 就可以調(diào)用 IlluminateSupportFacadesRoute 呢?這個(gè)就是別名的用途了,很多門面都有自己的別名,這樣我們就不必在代碼里面寫 use IlluminateSupportFacadesRoute,而是可以直接用 Route 了。

別名 Aliases

為什么我們可以在 larval 中全局用 Route,而不需要使用 use IlluminateSupportFacadesRoute?其實(shí)奧秘在于一個(gè) PHP 函數(shù):class_alias,它可以為任何類創(chuàng)建別名。larval 在啟動的時(shí)候?yàn)楦鱾€(gè)門面類調(diào)用了 class_alias 函數(shù),因此不必直接用類名,直接用別名即可。在 config 文件夾的 app 文件里面存放著門面與類名的映射:

"aliases" => [

    "App" => IlluminateSupportFacadesApp::class,
    "Artisan" => IlluminateSupportFacadesArtisan::class,
    "Auth" => IlluminateSupportFacadesAuth::class,
    ...
   ]

下面我們來看看 laravel 是如何為門面類創(chuàng)建別名的。

啟動別名Aliases服務(wù)

說到 larval 的啟動,我們離不開 index.php:

require __DIR__."/../bootstrap/autoload.php";

$app = require_once __DIR__."/../bootstrap/app.php";

$kernel = $app->make(IlluminateContractsHttpKernel::class);

$response = $kernel->handle(
  $request = IlluminateHttpRequest::capture()
);
...

第一句就是我們前面博客說的 composer 的自動加載,接下來第二句獲取 laravel 核心的 Ioc 容器,第三句“制造”出 Http 請求的內(nèi)核,第四句是我們這里的關(guān)鍵,這句牽扯很大,laravel 里面所有功能服務(wù)的注冊加載,乃至 Http 請求的構(gòu)造與傳遞都是這一句的功勞。

$request = IlluminateHttpRequest::capture()

這句是 laravel 通過全局 _SERVER 數(shù)組構(gòu)造一個(gè) Http 請求的語句,接下來會調(diào)用 Http 的內(nèi)核函數(shù) handle:

public function handle($request)
{
       try {
            $request->enableHttpMethodParameterOverride();

            $response = $this->sendRequestThroughRouter($request);
       } catch (Exception $e) {
            $this->reportException($e);
    
            $response = $this->renderException($request, $e);
       } catch (Throwable $e) {
            $this->reportException($e = new FatalThrowableError($e));

            $response = $this->renderException($request, $e);
       }

       event(new EventsRequestHandled($request, $response));
    
       return $response;
}

在 handle 函數(shù)方法中 enableHttpMethodParameterOverride 函數(shù)是允許在表單中使用 delete、put 等類型的請求。我們接著看 sendRequestThroughRouter:

protected function sendRequestThroughRouter($request)
{
    $this->app->instance("request", $request);

    Facade::clearResolvedInstance("request");

    $this->bootstrap();

    return (new Pipeline($this->app))
           ->send($request)
           ->through($this->app->shouldSkipMiddleware() ? [] : 
                                  $this->middleware)
           ->then($this->dispatchToRouter());
}

前兩句是在 larval 的 Ioc 容器設(shè)置 request 請求的對象實(shí)例,F(xiàn)acade 中清楚 request 的緩存實(shí)例。bootstrap:

public function bootstrap()
{
     if (! $this->app->hasBeenBootstrapped()) {
         $this->app->bootstrapWith($this->bootstrappers());
     }
}
        
protected $bootstrappers = [
            IlluminateFoundationBootstrapLoadEnvironmentVariables::class,
            IlluminateFoundationBootstrapLoadConfiguration::class,
            IlluminateFoundationBootstrapHandleExceptions::class,
            IlluminateFoundationBootstrapRegisterFacades::class,
            IlluminateFoundationBootstrapRegisterProviders::class,
            IlluminateFoundationBootstrapBootProviders::class,
];

$bootstrappers 是 Http 內(nèi)核里專門用于啟動的組件,bootstrap 函數(shù)中調(diào)用 Ioc 容器的 bootstrapWith 函數(shù)來創(chuàng)建這些組件并利用組件進(jìn)行啟動服務(wù)。app->bootstrapWith:

public function bootstrapWith(array $bootstrappers)
{
    $this->hasBeenBootstrapped = true;

    foreach ($bootstrappers as $bootstrapper) {
      $this["events"]->fire("bootstrapping: ".$bootstrapper, [$this]);

      $this->make($bootstrapper)->bootstrap($this);

      $this["events"]->fire("bootstrapped: ".$bootstrapper, [$this]);
    }
}

可以看到 bootstrapWith 函數(shù)也就是利用 Ioc 容器創(chuàng)建各個(gè)啟動服務(wù)的實(shí)例后,回調(diào)啟動自己的函數(shù) bootstrap,在這里我們只看我們 Facade 的啟動組件

IlluminateFoundationBootstrapRegisterFacades::class

RegisterFacades 的 bootstrap 函數(shù):

class RegisterFacades
{
    public function bootstrap(Application $app)
    {
        Facade::clearResolvedInstances();

        Facade::setFacadeApplication($app);

        AliasLoader::getInstance($app->make("config")->get("app.aliases", []))->register();
    }
}

可以看出來,bootstrap 做了一下幾件事:

清除了 Facade 中的緩存

設(shè)置 Facade 的 Ioc 容器

獲得我們前面講的 config 文件夾里面 app 文件 aliases 別名映射數(shù)組

使用 aliases 實(shí)例化初始化 AliasLoader

調(diào)用 AliasLoader->register()

public function register()
{
    if (! $this->registered) {
         $this->prependToLoaderStack();

         $this->registered = true;
    }
}

protected function prependToLoaderStack()
{
    spl_autoload_register([$this, "load"], true, true);
}

我們可以看出,別名服務(wù)的啟動關(guān)鍵就是這個(gè) spl_autoload_register,這個(gè)函數(shù)我們應(yīng)該很熟悉了,在自動加載中這個(gè)函數(shù)用于解析命名空間,在這里用于解析別名的真正類名。

別名 Aliases 服務(wù)

我們首先來看看被注冊到 spl_autoload_register 的函數(shù),load:

public function load($alias)
{
    if (static::$facadeNamespace && strpos($alias, static::$facadeNamespace) === 0) {
       $this->loadFacade($alias);

       return true;
    }

    if (isset($this->aliases[$alias])) {
         return class_alias($this->aliases[$alias], $alias);
    }
}

這個(gè)函數(shù)的下面很好理解,就是 class_alias 利用別名映射數(shù)組將別名映射到真正的門面類中去,但是上面這個(gè)是什么呢?實(shí)際上,這個(gè)是 laravel5.4 版本新出的功能叫做實(shí)時(shí)門面服務(wù)。

實(shí)時(shí)門面服務(wù)

其實(shí)門面功能已經(jīng)很簡單了,我們只需要定義一個(gè)類繼承 Facade 即可,但是 laravel5.4 打算更近一步——自動生成門面子類,這就是實(shí)時(shí)門面。
實(shí)時(shí)門面怎么用?看下面的例子:

namespace AppServices;

class PaymentGateway
{
    protected $tax;

    public function __construct(TaxCalculator $tax)
    {
        $this->tax = $tax;
    }
}

這是一個(gè)自定義的類,如果我們想要為這個(gè)類定義一個(gè)門面,在 laravel5.4 我們可以這么做:

use Facades {
    AppServicesPaymentGateway
};

Route::get("/pay/{amount}", function ($amount) {
    PaymentGateway::pay($amount);
});

那么這么做的原理是什么呢?我們接著看源碼:

protected static $facadeNamespace = "Facades";
if (static::$facadeNamespace && strpos($alias, static::$facadeNamespace) === 0) {
     $this->loadFacade($alias);

     return true;
}

如果命名空間是以 Facades 開頭的,那么就會調(diào)用實(shí)時(shí)門面的功能,調(diào)用 loadFacade 函數(shù):

protected function loadFacade($alias)
{
    tap($this->ensureFacadeExists($alias), function ($path) {
        require $path;
    });
}

tap 是 laravel 的全局幫助函數(shù),ensureFacadeExists 函數(shù)負(fù)責(zé)自動生成門面類,loadFacade 負(fù)責(zé)加載門面類:

protected function ensureFacadeExists($alias)
{
    if (file_exists($path = storage_path("framework/cache/facade-".sha1($alias).".php"))) {
        return $path;
    }

    file_put_contents($path, $this->formatFacadeStub(
                $alias, file_get_contents(__DIR__."/stubs/facade.stub")
            ));

    return $path;
}

可以看出來,laravel 框架生成的門面類會放到 stroge/framework/cache/ 文件夾下,名字以 facade 開頭,以命名空間的哈希結(jié)尾。如果存在這個(gè)文件就會返回,否則就要利用 file_put_contents 生成這個(gè)文件,formatFacadeStub:

protected function formatFacadeStub($alias, $stub)
{
    $replacements = [
        str_replace("/", "", dirname(str_replace("", "/", $alias))),
        class_basename($alias),
        substr($alias, strlen(static::$facadeNamespace)),
    ];

    return str_replace(
        ["DummyNamespace", "DummyClass", "DummyTarget"], $replacements, $stub
            );
}

簡單的說,對于 FacadesAppServicesPaymentGatewayreplacements 第一項(xiàng)是門面命名空間,將 FacadesAppServicesPaymentGateway 轉(zhuǎn)為 Facades/App/Services/PaymentGateway,取前面 Facades/App/Services/,再轉(zhuǎn)為命名空間 FacadesAppServices;第二項(xiàng)是門面類名,PaymentGateway;第三項(xiàng)是門面類的服務(wù)對象,AppServicesPaymentGateway,用這些來替換門面的模板文件:


替換后的文件是:


就是這么簡單?。?!

結(jié)語

門面的原理就是這些,相對來說門面服務(wù)的原理比較簡單,和自動加載相互配合使得代碼更加簡潔,希望大家可以更好的使用這些門面!

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/22924.html

相關(guān)文章

  • Laravel源碼解析之Model

    摘要:根據(jù)單一責(zé)任開發(fā)原則來講,在的開發(fā)過程中每個(gè)表都應(yīng)建立一個(gè)對外服務(wù)和調(diào)用。類似于這樣解析的數(shù)據(jù)操作分兩種它們除了有各自的特色外,基本的數(shù)據(jù)操作都是通過調(diào)用方法去完成整個(gè)。內(nèi)并沒有太多的代碼,大多都是處理數(shù)據(jù)庫鏈接。 showImg(https://segmentfault.com/img/bVbhjvY?w=600&h=296); 前言 提前預(yù)祝猿人們國慶快樂,吃好、喝好、玩好,我會在...

    CloudwiseAPM 評論0 收藏0
  • Laravel 服務(wù)提供者和門面模式

    摘要:服務(wù)提供者先看看定義服務(wù)提供者是所有應(yīng)用程序啟動的中心所在。通過本文,希望大家能夠了解服務(wù)提供者,,和實(shí)際調(diào)用的類的實(shí)例之間的關(guān)系。 以 Laravel 自帶的文件系統(tǒng)為例,在 config/app.php 的配置文件的 providers 數(shù)組中,注冊了一個(gè)服務(wù)提供者: IlluminateFilesystemFilesystemServiceProvider::class, 在 a...

    e10101 評論0 收藏0
  • PHP中的facade pattern(外觀模式)

    摘要:本文來自原文鏈接歡迎作客我們的學(xué)習(xí)群該篇屬于底層核心技術(shù)實(shí)戰(zhàn)揭秘這一課程底層核心概念解析這一章的擴(kuò)展閱讀??紤]到學(xué)員們的基礎(chǔ)差異,為了避免視頻當(dāng)中過于詳細(xì)而連篇累牘,故將一些底層實(shí)現(xiàn)相關(guān)的知識點(diǎn)以文章形式呈現(xiàn),供大家預(yù)習(xí)和隨時(shí)查閱。 本文來自pilishen.com----原文鏈接; 歡迎作客我們的php&Laravel學(xué)習(xí)群:109256050該篇屬于《Laravel底層核心技術(shù)實(shí)戰(zhàn)...

    jaysun 評論0 收藏0
  • Laravel中的核心概念

    摘要:可以為服務(wù)提供者的方法設(shè)置類型提示。方法將在所有其他服務(wù)提供者均已注冊之后調(diào)用。所有服務(wù)提供者都在配置文件中注冊??梢赃x擇推遲服務(wù)提供者的注冊,直到真正需要注冊綁定時(shí),這樣可以提供應(yīng)用程序的性能。 本文最早發(fā)布于 Rootrl的Blog 導(dǎo)言 Laravel是一款先進(jìn)的現(xiàn)代化框架,里面有一些概念非常重要。在上手Laravel之前,我認(rèn)為先弄懂這些概念是很有必要的。你甚至需要重溫下PHP...

    ddongjian0000 評論0 收藏0
  • 一個(gè) 16年畢業(yè)生所經(jīng)歷的 PHP 面試

    摘要:正確做法是給加索引,還有聯(lián)合索引,并不能避免全表掃描。 前言:有收獲的話請加顆小星星,沒有收獲的話可以 反對 沒有幫助 舉報(bào)三連 有心的同學(xué)應(yīng)該會看到我這個(gè)noteBook下面的其它知識,希望對你們有些許幫助。 本文地址 時(shí)間點(diǎn):2017-11 一個(gè)16年畢業(yè)生所經(jīng)歷的php面試 一、什么是面試 二、面試準(zhǔn)備 1. 問:什么時(shí)候開始準(zhǔn)備? 2. 問:怎么準(zhǔn)備? 三、面試...

    dabai 評論0 收藏0

發(fā)表評論

0條評論

wanghui

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<