laravel門面模式與服務提供者區別
以 laravel 自帶的文件系統為例,在 config/app.php 的配置文件的 providers 數組中,注冊了一個服務提供者:
IlluminateFilesystemFilesystemServiceProvider::class,
在 alias 數組中定義了一個門面:
‘File’?=>?IlluminateSupportFacadesFile::class,
通過這兩個步驟,我們就可以非常方便的使用 Laravel 提供的文件系統相關的操作,而且調用形式很簡潔,如:
File::exist?($path),判斷文件是否存在。 File::get?($path,?$lock?=?false),獲取一個文件的內容。 File::append?($path,?$data),把內容追加到一個文件末尾。 File::files?($directory),獲取一個目錄下所有文件。
那么這是如何做到的呢?下面分別講一講 Laravel 的服務提供者和門面模式。
服務提供者
先看看定義:
服務提供者是所有 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,和實際調用的類的實例之間的關系。
更多Laravel相關技術文章,請訪問Laravel框架入門教程欄目進行學習!