Laravel 服務提供者和門面模式

以 Laravel 自帶的文件系統爲例,在 config/app.php 的配置文件的 providers 數組中,註冊了一個服務提供者:php

IlluminateFilesystemFilesystemServiceProvider::class,laravel

在 alias 數組中定義了一個門面:數組

‘File’ => IlluminateSupportFacadesFile::class,app

經過這兩個步驟,咱們就能夠很是方便的使用 Laravel 提供的文件系統相關的操做,並且調用形式很簡潔,如:ide

File::exist($path),判斷文件是否存在。測試

File::get($path, $lock = false),獲取一個文件的內容。this

File::append($path, $data),把內容追加到一個文件末尾。代理

File::files($directory),獲取一個目錄下全部文件。code

那麼這是如何作到的呢?下面分別講一講 Laravel 的服務提供者和門面模式。component

服務提供者

先看看定義:

服務提供者是全部 Laravel 應用程序啓動的中心所在。包括你本身的應用程序,以及全部的 Laravel 核心服務,都是經過服務提供者啓動的。

在文件系統這個服務提供者中,位置 /vendor/laravel/framework/src/Illuminate/Filesystem/FilesystemServiceProvider.php,register 方法能夠看到綁定了一個單例:

protected function registerNativeFilesystem()
{
    $this->app->singleton('files', function () {
        return new Filesystem;
    });
}

這個單例是 Filesystem 這個類的單例模式。固然,這個服務提供者中也能夠綁定其餘的單例,或作更多的事情。咱們這裏只研究 File::exist() 這種調用方式的原理。

那麼這樣一來就有個 files 的單例,其實是 Filesystem 這個類的實例。

此時,若是沒有 Facade,也是能夠調用到 Filesystem 這個實例的方法的,那就是這樣調用:

app(‘files’)->exist($path)

好了,如今開始講 Facade.

Facade 門面模式

先看下簡介:

Facades /fəˈsäd/ 爲應用程序的服務容器中可用的類提供了一個「靜態」接口。Laravel 自帶了許多的 facades,能夠用來訪問其幾乎全部的服務。Laravel facades 就是服務容器裏那些基類的「靜態代理」,相比於傳統的靜態方法調用,facades 在提供更簡潔且豐富的語法的同時,還有更好的可測試性和擴展性。

本文一開始講到 alias 數組定義了一個 File,具體的類是

IlluminateSupportFacadesFile::class,

它的內容是:

class File extends Facade
{
    /**
     * Get the registered name of the component.
     *
     * @return string
     */
    protected static function getFacadeAccessor()
    {
        return 'files';
    }
}

它實際上返回了一個名稱,注意這個名稱 files,不就是剛剛綁定的單例模式的名稱嗎?沒錯。

這樣一來,就可使用 File 這個別名或者說門面,來調用這個 Filesystem 實例中的方法了。

經過本文,但願你們可以瞭解服務提供者,Facade,和實際調用的類的實例之間的關係。

(原文地址:https://blog.tanteng.me/2017/...

相關文章
相關標籤/搜索