綁定基礎
幾乎所有的服務容器綁定都是在 服務提供者 中完成。
在目錄結(jié)構如下圖
![](http://img.jbzj.com/file_images/article/202006/202061495950298.jpg?202051495957)
注:如果一個類沒有基于任何接口那么就沒有必要將其綁定到容器。容器并不需要被告知如何構建對象,因為它會使用 PHP 的反射服務自動解析出具體的對象。
簡單的綁定
在一個服務提供者中,可以通過 $this->app 變量訪問容器,然后使用 bind 方法注冊一個綁定,該方法需要兩個參數(shù),第一個參數(shù)是我們想要注冊的類名或接口名稱,第二個參數(shù)是返回類的實例的閉包:
$this->app->bind('HelpSpot\API', function ($app) {
return new HelpSpot\API($app->make('HttpClient'));
});
注意到我們將容器本身作為解析器的一個參數(shù),然后我們可以使用該容器來解析我們正在構建的對象的子依賴。
綁定一個單例
singleton 方法綁定一個只會解析一次的類或接口到容器,然后接下來對容器的調(diào)用將會返回同一個對象實例:
$this->app->singleton('HelpSpot\API', function ($app) {
return new HelpSpot\API($app->make('HttpClient'));
});
綁定原始值
你可能有一個接收注入類的類,同時需要注入一個原生的數(shù)值比如整型,可以結(jié)合上下文輕松注入這個類需要的任何值:
$this->app->when('App\Http\Controllers\UserController')
->needs('$variableName')
->give($value);
綁定接口到實現(xiàn)
服務容器的一個非常強大的功能是其綁定接口到實現(xiàn)。我們假設有一個 EventPusher 接口及其實現(xiàn)類 RedisEventPusher ,編寫完該接口的 RedisEventPusher 實現(xiàn)后,就可以將其注冊到服務容器:
$this->app->bind(
'App\Contracts\EventPusher',
'App\Services\RedisEventPusher'
);
這段代碼告訴容器當一個類需要 EventPusher 的實現(xiàn)時將會注入 RedisEventPusher,現(xiàn)在我們可以在構造器或者任何其它通過服務容器注入依賴的地方進行 EventPusher 接口的依賴注入:
use App\Contracts\EventPusher;
/**
* 創(chuàng)建一個新的類實例
*
* @param EventPusher $pusher
* @return void
*/
public function __construct(EventPusher $pusher){
$this->pusher = $pusher;
}
上下文綁定
有時侯我們可能有兩個類使用同一個接口,但我們希望在每個類中注入不同實現(xiàn),例如,兩個控制器依賴 Illuminate\Contracts\Filesystem\Filesystem 契約的不同實現(xiàn)。Laravel 為此定義了簡單、平滑的接口:
use Illuminate\Support\Facades\Storage;
use App\Http\Controllers\VideoController;
use App\Http\Controllers\PhotoControllers;
use Illuminate\Contracts\Filesystem\Filesystem;
$this->app->when(PhotoController::class)
->needs(Filesystem::class)
->give(function () {
return Storage::disk('local');
});
$this->app->when(VideoController::class)
->needs(Filesystem::class)
->give(function () {
return Storage::disk('s3');
});
標簽
少數(shù)情況下,我們需要解析特定分類下的所有綁定,例如,你正在構建一個接收多個不同 Report 接口實現(xiàn)的報告聚合器,在注冊完 Report 實現(xiàn)之后,可以通過 tag 方法給它們分配一個標簽:
$this->app->bind('SpeedReport', function () {
//
});
$this->app->bind('MemoryReport', function () {
//
});
$this->app->tag(['SpeedReport', 'MemoryReport'], 'reports');
這些服務被打上標簽后,可以通過 tagged 方法來輕松解析它們:
$this->app->bind('ReportAggregator', function ($app) {
return new ReportAggregator($app->tagged('reports'));
});
擴展綁定
extend 方法允許對解析服務進行修改。例如,當服務被解析后,可以運行額外代碼裝飾或配置該服務。extend 方法接收一個閉包來返回修改后的服務:
$this->app->extend(Service::class, function($service) {
return new DecoratedService($service);
});
總結(jié)
到此這篇關于Laravel服務容器綁定的文章就介紹到這了,更多相關Laravel服務容器綁定內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
您可能感興趣的文章:- 詳解Laravel服務容器的綁定與解析
- 詳解如何實現(xiàn)Laravel的服務容器的方法示例
- laravel ajax curd 搜索登錄判斷功能的實現(xiàn)
- Laravel中Kafka的使用詳解
- laravel使用redis隊列實例講解
- Laravel的加密解密與哈希實例講解
- Laravel中10個有用的用法小結(jié)
- 詳解Laravel服務容器的優(yōu)勢