这两个概念对于 Laravel 的使用者来讲应该并不陌生,尤为是当你但愿扩展或者替换 Laravel 核心库的时候,理解和合理使用它们能够极大提高 Laravel 的战斗力。这里以建立一个本身的 ServiceProvider 为例理解 Inversion of Control 和 Facade 在 Laravel 中的应用。php
控制反转(Inversion of Control,缩写为IoC),是面向对象编程中的一种设计原则,能够用来减低计算机代码之间的耦合度。其中最多见的方式叫作依赖注入(Dependency Injection,简称DI),还有一种方式叫“依赖查找”(Dependency Lookup)。经过控制反转,对象在被建立的时候,由一个调控系统内全部对象的外界实体,将其所依赖的对象的引用传递给它。 — 维基百科html
简单说来,就是一个类把本身的的控制权交给另一个对象,类间的依赖由这个对象去解决。依赖注入属于依赖的显示申明,而依赖查找则是经过查找来解决依赖。laravel
注入一个类:编程
App::bind('foo', function($app) { return new FooBar; });
这个例子的意思是建立一个别名为 foo 的类,使用时实际实例化的是 FooBar。数组
使用这个类的方法是:app
$value = App::make('foo');
$value 其实是 FooBar 对象。ide
若是但愿使用单例模式来实例化类,那么使用:this
App::singleton('foo', function() { return new FooBar; });
这样的话每次实例化后的都是同一个对象。spa
注入类的更多例子能够看 Laravel 官网设计
你可能会疑问上面的代码应该写在哪儿呢?答案是你但愿他们在哪儿运行就写在哪儿。0 —— 0 知道写哪儿还用来看这种基础文章么!
为了让依赖注入的代码不至于写乱,Laravel 搞了一个 服务提供器(Service Provider)的东西,它将这些依赖汇集在了一块,统一申明和管理,让依赖变得更加容易维护。
定义一个服务提供器:
use Illuminate\Support\ServiceProvider; class FooServiceProvider extends ServiceProvider { public function register() { $this->app->bind('foo', function() { return new Foo; }); } }
这个代码也不难理解,就是申明一个服务提供器,这个服务提供器有一个 register的方法。这个方法实现了咱们上面讲到的依赖注入。
当咱们执行下面代码:
App::register('FooServiceProvider');
咱们就完成一个注入了。可是这个仍是得手动写,因此怎么让 Laravel 本身来作这事儿呢?
咱们只要在 app/config/app.php 中的 providers 数组里面增长一行:
'providers' => [ … ‘FooServiceProvider’, ],
这样咱们就能够使用 App::make(‘foo’) 来实例化一个类了。
你不由要问了,这么写也太难看了吧?莫慌,有办法。
为了让 Laravel 中的核心类使用起来更加方便,Laravel实现了门面模式。
外觀模式(Facade pattern),是軟件工程中经常使用的一種軟件設計模式,它為子系統中的一組接口提供一個統一的高層接口,使得子系統更容易使用。 — 维基百科
咱们使用的大部分核心类都是基于门面模式实现的。例如:
$value = Cache::get('key');
这些静态调用实际上调用的并非静态方法,而是经过 PHP 的魔术方法__callStatic() 讲请求转到了相应的方法上。
那么如何讲咱们前面写的服务提供器也这样使用呢?方法很简单,只要这么写:
use Illuminate\Support\Facades\Facade; class Foo extends Facade { protected static function getFacadeAccessor() { return ‘foo’; } }
这样咱们就能够经过 Foo::test() 来调用咱们以前真正的 FooBar 类的方法了。
有时候咱们可能将 Facade 放在咱们扩展库中,它有比较深的命名空间,如:\Library\MyClass\Foo。这样致使使用起来并不方便。Laravel 能够用别名来替换掉这么长的名字。
咱们只要在 app/config/app.php 中 aliases 下增长一行便可:
'aliases' => [ … 'Foo' => ‘Library\MyClass\Foo’, ],
这样它的使用就由 \Library\MyClass\Foo::test() 变成 Foo::test() 了。
因此有了控制反转(Inversion of Control)和门面模式(Facade),实际还有服务提供器(Service Providers)和别名(Alias),咱们建立本身的类库和扩展 Laravel 都会方便不少。
这里总结一下建立本身类库的方法: