Laravel工程化项目三:核心概念:ServiceProvider

2022-10-11 21:50:49 172 0
魁首哥

上一节:

介绍

ServiceProvider是Laravel框架的核心机制之一,也是它的组件规范,框架的启动初始化重要的一步是ServiceProvider的初始化,ServiceProvider主要用于组件的启动类的类的注入及实例化。

查看vendor/laravel/framework/src/Illuminate下的源代码会发现,基本上每个组件都有一个xxServiceProvider的类。

例如: verndor/laravel/framework/src/Illuminate/Log/LogServiceProvider.php

 class LogServiceProvider extends ServiceProvider
{
    /**
     * Register the service provider.
     *
     * @return void
     */
    public function register()
    {
        $this->app->singleton('log', function ($app) {
            return new LogManager($app);
        });
    }
}
  

这里register方法完成log跟LogManager的绑定工作,也就是我们对log的操作实际是对LogManager的操作,当我们在程序中可以这样使用:

 // 这只是个例子,不推荐这样使用
 app('log')->info('this is test msg', []);
  

以上就完成了日志的记录。

原理

Laravel在框架的Bootstrap阶段通过 src/Illuminate/Foundation/Bootstrap/RegisterProvider.php、BootProviders.php。 两个类完成ServiceProvider的注册、启动工作:

 public function registerConfiguredProviders()
{
    $providers = Collection::make($this->make('config')->get('app.providers'))
                    ->partition(function ($provider) {
                        return strpos($provider, 'Illuminate\\') === 0;
                    });

    $providers->splice(1, 0, [$this->make(PackageManifest::class)->providers()]);

    (new ProviderRepository($this, new Filesystem, $this->getCachedServicesPath()))
                ->load($providers->collapse()->toArray());
}

public function boot()
{
    if ($this->isBooted()) {
        return;
    }

    // Once the application has booted we will also fire some "booted" callbacks
    // for any listeners that need to do work after this initial booting gets
    // finished. This is useful when ordering the boot-up processes we run.
    $this->fireAppCallbacks($this->bootingCallbacks);

    array_walk($this->serviceProviders, function ($p) {
        $this->bootProvider($p);
    });

    $this->booted = true;

    $this->fireAppCallbacks($this->bootedCallbacks);
}
  

框架会扫描config/app.php中注册的ServiceProvider并依次执行它们的register()、boot() 方法,对于上面的LogServiceProvider来说,就执行了 LogManager与log的绑定工作了。类比的可以看下QueueServiceProvider.php RedisServiceProvider.php 完成了哪些工作。

自定义ServiceProvider

Laravel的ServiceProvider在app/config/app.php中引入,如果不需要的可以去掉。

举栗子:现在我们需要一个检测项目.env文件是否存在的功能,可以通过自定义一个EnvCheckServiceProvider来做,因为这个工作需要在框架请求处理之前完成。

  • 新增ServiceProvider
 php artisan make:provider EnvCheckServiceProvider
  
 class EnvCheckServiceProvider extends ServiceProvider
{
    /**
     * Register services.
     *
     * @return void
     */
    public function register()
    {
        if (!file_exists(base_path('.env'))) {
            throw new \Exception(".env file not found. Please add this file. ");
        }
    }

    /**
     * Bootstrap services.
     *
     * @return void
     */
    public function boot()
    {
        //
    }
}
  
  • 在config/app.php 中加入 EnvCheckServiceProvider.php
 
'providers' => [
    //xxxxx其它provider
    \App\Providers\EnvCheckServiceProvider::class,

]

  

我们测试把目录下的.env删除掉,再访问之前的接口就会发现报错了

ServiceProvider的使用场景

当你的功能需要在请求处理前就要完成的时候,比如某个类的实例化、某个请求环境的检测、某个文件的创建等等,可以使用ServiceProvider来完成。

文末福利推荐

  • 本文代码:
  • 参考文档:
  • 外卖领券,感谢关注

收藏
分享
海报
0 条评论
172
上一篇:PHP函数import_request_variables()用法及实例分析 下一篇:必知的php数组函数

本站已关闭游客评论,请登录或者注册后再评论吧~

忘记密码?

图形验证码