因实际项目需求,近日在开发 laravel-database-logger 包的时候,发现设置 ServiceProvider defer
属性设置为 true
时,会致使在register
方法中注册的 middleware
无效。php
class ServiceProvider extends \Illuminate\Support\ServiceProvider { protected $defer = true; public function register() { $this->mergeConfigFrom( __DIR__ . '/../config/config.php', 'ibrand.dblogger' ); $this->app->singleton(DbLogger::class, function ($app) { return new DbLogger(); }); //当 $defer 设置为 true 时,在路由中引用 databaselogger middleware 会报错,提示 databaselogger class not found. $this->app[\Illuminate\Routing\Router::class]->middleware('databaselogger', Middleware::class); } public function provides() { return [DbLogger::class]; } }
当问题出现的时候就怀疑是由于设置了 defer
属性设置为 true
致使的,马上就修改源码把 protected $defer = true;
的代码注释掉,结果仍然是提示 databaselogger class not found.
,说明 Laravel 并无注册此 ServiceProvder
laravel
接下来就是想如何解决此问题,尝试了下面的方法:git
1. 验证自己代码是否存在问题github
在正常注册的 AppServiceProvider
中注册本身的 ServiceProvider
bootstrap
public function register() { // $this->app->register(\Ibrand\DatabaseLogger\ServiceProvider::class); }
注册后结果一切正常。缓存
2. 研究源码
在 config/app.php
中 providers
注册无效,可是在其余 ServiceProvider
中注册有效,说明是其余问题。app
经过研究 Illuminate\Foundation\Application
源码找到 registerConfiguredProviders
方法:ide
Laravel 是在此方法中去读取 config/app.php
中的 providers
内容并load
到 ProviderRepository
中。this
(new ProviderRepository($this, new Filesystem, $this->getCachedServicesPath())) ->load($providers->collapse()->toArray());
重点在 $this->getCachedServicesPath()
,经过源码发现 Laravel 是根据 bootstrap/cache/services.php
文件去决定如何注册ServiceProvider
。code
此时想到了为何以前注释了 //protected $defer = true;
代码后仍然无效的缘由。
因此为了让注释后的 //protected $defer = true;
代码有效须要执行
php artisan clear-compiled php artisan optimize
以后问题就解决了,也更加深刻理解了 ServiceProvider 的原理。
因此切记:若是准备采用延迟加载
ServiceProvider
时,严禁进行注册 middleware, route 等系列操做。同时,更改defer
属性值后,须要执行php artisan clear-compiled
和php artisan optimize
以更新 ServiceProvider 缓存。
3. 为何 AppServiceProvider 中注册有效?
愿意很简单,由于 AppServiceProvider
并无延迟加载,所以在执行 AppServiceProvider
中 register
方法去注册新的ServiceProvider
也是不会延迟加载的。
ServiceProvider
defer
属性值后,须要执行 php artisan clear-compiled
和 php artisan optimize
以更新 ServiceProvider 缓存。ServiceProvider
注册 middleware
和 route
。