服务提供者


1、简介

服务提供者是所有Laravel应用启动的中心,你自己的应用以及所有Laravel的核心服务都是通过服务提供者启动。

但是,我们所谓的“启动”指的是什么?通常,这意味着注册事物,包括注册服务容器绑定、事件监听器、中间件甚至路由。服务提供者是应用配置的中心。

如果你打开Laravel自带的config/app.php文件,将会看到一个providers数组,这里就是应用所要加载的所有服务提供者类,当然,其中很多是延迟加载的,也就是说不是每次请求都会被加载,只有真的用到它们的时候才会加载。

本章里你将会学习如何编写自己的服务提供者并在Laravel应用中注册它们。

2、编写服务提供者

所有的服务提供者继承自Illuminate\Support\ServiceProvider类。继承该抽象类要求至少在服务提供者中定义一个方法:register。在register方法内,你唯一要做的事情就是绑事物到服务容器,不要尝试在其中注册任何时间监听器,路由或者任何其它功能。

通过Artisan命令make:provider可以简单生成一个新的提供者:

  1. php artisan make:provider RiakServiceProvider

register方法

正如前面所提到的,在register方法中只绑定事物到服务容器,而不要做其他事情,否则话,一不小心就能用到一个尚未被加载的服务提供者提供的服务。

现在让我们来看看一个基本的服务提供者长什么样:

  1. <?php
  2. namespace App\Providers;
  3. use Riak\Connection;
  4. use Illuminate\Support\ServiceProvider;
  5. class RiakServiceProvider extends ServiceProvider{
  6. /**
  7. * 在容器中注册绑定.
  8. *
  9. * @return void
  10. */
  11. public function register()
  12. {
  13. $this->app->singleton('Riak\Contracts\Connection', function ($app) {
  14. return new Connection(config('riak'));
  15. });
  16. }
  17. }

该服务提供者只定义了一个register方法,并使用该方法在服务容器中定义了一个Riak\Contracts\Connection的实现。

boot方法

如果我们想要在服务提供者中注册视图composer该怎么做?这就要用到boot方法了。该方法在所有服务提供者被注册以后才会被调用,这就是说我们可以在其中访问框架已注册的所有其它服务:

  1. <?php
  2. namespace App\Providers;
  3. use Illuminate\Support\ServiceProvider;
  4. class EventServiceProvider extends ServiceProvider{
  5. /**
  6. * Perform post-registration booting of services.
  7. *
  8. * @return void
  9. */
  10. public function boot()
  11. {
  12. view()->composer('view', function () {
  13. //
  14. });
  15. }
  16. /**
  17. * 在容器中注册绑定.
  18. *
  19. * @return void
  20. */
  21. public function register()
  22. {
  23. //
  24. }
  25. }

boot方法的依赖注入

我们可以在boot方法中类型提示依赖,服务容器会自动注册你所需要的依赖:

  1. use Illuminate\Contracts\Routing\ResponseFactory;
  2. public function boot(ResponseFactory $factory){
  3. $factory->macro('caps', function ($value) {
  4. //
  5. });
  6. }

3、注册服务提供者

所有服务提供者都是通过配置文件config/app.php中进行注册,该文件包含了一个列出所有服务提供者名字的providers数组,默认情况下,其中列出了所有核心服务提供者,这些服务提供者启动Laravel核心组件,比如邮件、队列、缓存等等。

要注册你自己的服务提供者,只需要将其追加到该数组中即可:

  1. 'providers' => [
  2. // 其它服务提供者
  3. App\Providers\AppServiceProvider::class,
  4. ],

4、延迟加载服务提供者

如果你的提供者仅仅只是在服务容器中注册绑定,你可以选在延迟加载该绑定直到注册绑定真的需要时再加载,延迟加载这样的一个提供者将会提升应用的性能,因为它不会在每次请求时都从文件系统加载。

想要延迟加载一个提供者,设置defer属性为true并定义一个provides方法,该方法返回该提供者注册的服务容器绑定:

  1. <?php
  2. namespace App\Providers;
  3. use Riak\Connection;
  4. use Illuminate\Support\ServiceProvider;
  5. class RiakServiceProvider extends ServiceProvider{
  6. /**
  7. * 服务提供者加是否延迟加载.
  8. *
  9. * @var bool
  10. */
  11. protected $defer = true;
  12. /**
  13. * 注册服务提供者
  14. *
  15. * @return void
  16. */
  17. public function register()
  18. {
  19. $this->app->singleton('Riak\Contracts\Connection', function ($app) {
  20. return new Connection($app['config']['riak']);
  21. });
  22. }
  23. /**
  24. * 获取由提供者提供的服务.
  25. *
  26. * @return array
  27. */
  28. public function provides()
  29. {
  30. return ['Riak\Contracts\Connection'];
  31. }
  32. }

Laravel 编译并保存所有延迟服务提供者提供的服务及服务提供者的类名。然后,只有当你尝试解析其中某个服务时Laravel才会加载其服务提供者。