首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

Laravel神奇的服务容器

Laravel的核心是IocContainer, 文档中称其为“服务容器”,服务容器是一个用于管理类依赖和执行依赖注入的强大工具,Laravel中的功能模块比如 Route、Eloquent ORM、Request、Response等等等等,实际上都是与核心无关的类模块提供的,这些类从注册到实例化,最终被我们所使用,其实都是 laravel 的服务容器负责的。

服务容器中有两个概念控制反转(IOC)和依赖注入(DI):

依赖注入和控制反转是对同一件事情的不同描述,它们描述的角度不同。依赖注入是从应用程序的角度在描述,应用程序依赖容器创建并注入它所需要的外部资源。而控制反转是从容器的角度在描述,容器控制应用程序,由容器反向的向应用程序注入应用程序所需要的外部资源。

在Laravel中框架把自带的各种服务绑定到服务容器,我们也可以绑定自定义服务到容器。当应用程序需要使用某一个服务时,服务容器会讲服务解析出来同时自动解决服务之间的依赖然后交给应用程序使用。

本篇就来探讨一下Laravel中的服务绑定和解析是如何实现的

服务绑定

常用的绑定服务到容器的方法有instance, bind, singleton, alias。下面我们分别来看一下。

instance

将一个已存在的对象绑定到服务容器里,随后通过名称解析该服务时,容器将总返回这个绑定的实例。

代码语言:javascript
复制
$api = new HelpSpot\API(new HttpClient);
$this->app->instance('HelpSpot\Api', $api);

会把对象注册到服务容器的$instances属性里

代码语言:javascript
复制
[
    'HelpSpot\Api' => $api//$api是API类的对象,这里简写了
]

bind

绑定服务到服务容器

有三种绑定方式:

代码语言:javascript
复制
1.绑定自身
$this->app->bind('HelpSpot\API', null);

2.绑定闭包
$this->app->bind('HelpSpot\API', function () {
   return new HelpSpot\API();
});//闭包直接提供类实现方式
$this->app->bind('HelpSpot\API', function ($app) {
   return new HelpSpot\API($app->make('HttpClient'));
});//闭包返回需要依赖注入的类
3. 绑定接口和实现
$this->app->bind('Illuminate\Tests\Container\IContainerContractStub', 'Illuminate\Tests\Container\ContainerImplementationStub');

bind把服务注册到服务容器的$bindings属性里类似这样:

代码语言:javascript
复制
$bindings = [
   'HelpSpot\API' =>  [//闭包绑定
      'concrete' => function ($app, $paramters = []) {
         return $app->build('HelpSpot\API');
      },
      'shared' => false//如果是singleton绑定,这个值为true
   ]    
   'Illuminate\Tests\Container\IContainerContractStub' => [//接口实现绑定
      'concrete' => 'Illuminate\Tests\Container\ContainerImplementationStub',
      'shared' => false
   ]
]

singleton

代码语言:javascript
复制
public function singleton($abstract, $concrete = null)
{
   $this->bind($abstract, $concrete, true);
}

singleton 方法是bind方法的变种,绑定一个只需要解析一次的类或接口到容器,然后接下来对于容器的调用该服务将会返回同一个实例

alias

把服务和服务别名注册到容器:

代码语言:javascript
复制
public function alias($abstract, $alias)
{
   $this->aliases[$alias] = $this->normalize($abstract);
}

alias 方法在上面讲bind方法里有用到过,它会把把服务别名和服务类的对应关系注册到服务容器的$aliases属性里。例如: $this->app->alias('\Illuminate\ServiceName', 'service_alias'); 绑定完服务后在使用时就可以通过 $this->app->make('service_alias'); 将服务对象解析出来,这样make的时候就不用写那些比较长的类名称了,对make方法的使用体验上有很大提升。

服务解析

服务解析是从服务容器中解析出在服务绑定过程中注册到服务容器里的对象,我们能过很方便的通过make方法来解析出服务对象,通过如果服务对象如果有依赖其他类对象的话服务容器也会帮助我们依次解决依赖从而免去了在程序里使用服务对象前先要在代码里实力化一堆依赖对象的繁重操作。 你可以点击“阅读原文”查看Laravel服务容器详细的源码解析,希望看完后能解决你在使用过程中的一些疑惑。

下一篇
举报
领券