前言
Laravel 是一款 PHP 开源框架,最近学习了一下 Symfony,现在来了解一下 Laravel 的最新版本的一些东西。
其实说到服务器容器,相信大家都会直接提到依赖注入的概念,其实服务容器的概念,就和我们设计模式中的对象池差不多,把所有的对象都放在一个池子里面去,而不必一个个去 new 了。而且支持每次都新建和单例,等等。
在这些的基础之上,衍生出了 2 个概念Ioc:控制反转,Di:依赖注入,这些概念,在我的认知里,早起出自 java 框架之中。主要的目的就是实现对象依赖解耦。具体的设计模式的理念,可以参考我博客中的设计模式一系列的文章。
下述主要参考了谋篇 laravel 服务容器介绍摘录+部分自我实践整理。对比了一下和官网的介绍差不多,更多的主要是例子的说明。
正题
Laravel 中有一大堆访问 Container 实例的姿势,比如最简单的:
| 1 | $container = app(); | 
但我们还是先关注下 Container 类本身。
| 1 | Laravel 官方文档中一般使用 $this->app 代替 $container。它是 Application 类的实例,而 Application 类继承自 Container 类。 | 
用法一:基本用法,用type hint (类型提示) 注入 依赖:
| 1 | 
 | 
接下来用 Container 的 make 方法来代替 new MyClass:
| 1 | $instance = $container->make(MyClass::class); | 
Container 会自动实例化依赖的对象,所以它等同于:
| 1 | $instance = new MyClass(new AnotherClass()); | 
如果 AnotherClass 也有 依赖,那么 Container 会递归注入它所需的依赖。
Container 使用 Reflection (反射) 来找到并实例化构造函数参数中的那些类。
用法二:Binding Interfaces to Implementations (绑定接口到实现)
用 Container 可以轻松地写一个接口,然后在运行时实例化一个具体的实例。 首先定义接口:
| 1 | interface MyInterface { /* ... */ } | 
然后声明实现这些接口的具体类。下面这个类不但实现了一个接口,还依赖了实现另一个接口的类实例:
| 1 | class MyClass implements MyInterface | 
现在用 Container 的 bind() 方法来让每个 接口 和实现它的类一一对应起来:
| 1 | $container->bind(MyInterface::class, MyClass::class); | 
最后,用接口名 而不是 类名 来传给 make():
| 1 | $instance = $container->make(MyInterface::class); | 
注意:如果你忘记绑定它们,会导致一个 Fatal Error:”Uncaught ReflectionException: Class MyInterface does not exist”。
实战
下面是可封装的 Cache 层:
| 1 | interface Cache | 
用法三:Binding Abstract & Concret Classes (绑定抽象类和具体类)
绑定还可以用在抽象类:
| 1 | $container->bind(MyAbstract::class, MyConcreteClass::class); | 
或者继承的类中:
| 1 | $container->bind(MySQLDatabase::class, CustomMySQLDatabase::class); | 
用法四:自定义绑定
如果类需要一些附加的配置项,可以把 bind() 方法中的第二个参数换成 Closure (闭包函数):
| 1 | $container->bind(Database::class, function (Container $container) { | 
闭包也可用于定制 具体类 的实例化方式:
| 1 | $container->bind(GitHub\Client::class, function (Container $container) { | 
用法五:Resolving Callbacks (回调)
可用 resolveing()方法来注册一个 callback (回调函数),而不是直接覆盖掉之前的 绑定。 这个函数会在绑定的类解析完成之后调用。
注意此时的回调函数中,第一个参数是对应被解析的对象,第二个参数是容器(container)<=>应用(app).
| 1 | $container->resolving(GitHub\Client::class, function ($client, Container $container) { | 
如果有一大堆 callbacks,他们全部都会被调用。对于 接口 和 抽象类 也可以这么用:
| 1 | $container->resolving(Logger::class, function (Logger $logger) { | 
更 diao 的是,还可以注册成「什么类解析完之后都调用」:
| 1 | $container->resolving(function ($object, Container $container) { | 
但这个估计只有
logging和debugging才会用到。
用法六:Extending a Class (扩展一个类)
使用 extend() 方法,可以封装一个类然后返回一个不同的对象 (代理模式):
(为什么不是装饰器模式?,因为装饰器模式不需要实现一样的接口,但是代理模式下,代理类需要和原来的类一样实现同一接口).
| 1 | $container->extend(APIClient::class, function ($client, Container $container) { | 
注意:这两个类要实现相同的 接口,不然用类型提示的时候会出错:.
| 1 | interface Getable | 
用法七:单例
使用 bind() 方法绑定后,每次解析时都会新实例化一个对象(或重新调用闭包),如果想获取 单例 ,则用 singleton() 方法代替 bind():
| 1 | $container->singleton(Cache::class, RedisCache::class); | 
绑定单例 闭包
| 1 | $container->singleton(Database::class, function (Container $container) { | 
绑定 具体类 的时候,不需要第二个参数:
| 1 | $container->singleton(MySQLDatabase::class); | 
在每种情况下,单例 对象将在第一次需要时创建,然后在后续重复使用。
如果你已经有一个 实例 并且想重复使用,可以用 instance() 方法。
| 1 | $container->instance(Container::class, $container); | 
Laravel 就是用这种方法来确保每次获取到的都是同一个 Container 实例:
用法七:Arbitrary Binding Names (任意绑定名称)
Container 还可以绑定任意字符串而不是 类/接口名称。但这种情况下不能使用类型提示,并且只能用 make() 来获取实例。
| 1 | $container->bind('database', MySQLDatabase::class); | 
为了同时支持类/接口名称和短名称,可以使用 alias():
| 1 | $container->singleton(Cache::class, RedisCache::class); | 
用法八:保存任何值
Container 还可以用来保存任何值,例如 configuration 数据:
| 1 | $container->instance('database.name', 'testdb'); | 
它支持数组访问语法,这样用起来更自然:
| 1 | $container['database.name'] = 'testdb'; | 
这是因为 Container 实现了 PHP 的 ArrayAccess 接口。
当处理 Closure 绑定的时候,你会发现这个方式非常好用:
| 1 | $container->singleton('database', function (Container $container) { | 
Laravel 自己没有用这种方式来处理配置项,它使用了一个单独的 Config 类本身。 PHP-DI 用了。
数组访问语法还可以代替 make() 来实例化对象:.
| 1 | $db = $container['database']; | 
用法九:Dependency Injection for Functions & Methods (给函数或方法注入依赖)
除了给构造函数注入依赖,Laravel 还可以往任意函数中注入:
| 1 | function do_something(Cache $cache) { /* ... */ } | 
函数的附加参数可以作为索引或关联数组传递:
| 1 | function show_product(Cache $cache, $id, $tab = 'details') { /* ... */ } | 
除此之外,闭包:
| 1 | $closure = function (Cache $cache) { /* ... */ }; | 
静态方法:
| 1 | class SomeClass | 
实例的方法:
| 1 | class PostController | 
都可以注入。
用法十: 调用实例方法的快捷方式
使用 ClassName@methodName 语法可以快捷调用实例中的方法:
| 1 | $container->call('PostController@index'); | 
因为 Container 被用来实例化类。意味着:
依赖 被注入进构造函数(或者方法);
如果需要复用实例,可以定义为单例;
可以用接口或任何名称来代替具体类。
所以这样调用也可以生效:
| 1 | class PostController | 
最后,还可以传一个「默认方法」作为第三个参数。如果第一个参数是没有指定方法的类名称,则将调用默认方法。 Laravel 用这种方式来处理 event handlers:
| 1 | $container->call(MyEventHandler::class, $parameters, 'handle'); | 
用法十一:Method Call Bindings (方法调用绑定)
bindMethod() 方法可用来覆盖方法,例如用来传递其他参数:
| 1 | $container->bindMethod('PostController@index', function ($controller, $container) { | 
下面的方式都有效,调用闭包来代替调用原始的方法:
| 1 | $container->call('PostController@index'); | 
但是,call() 的任何其他参数都不会传递到闭包中,因此不能使用它们。
| 1 | $container->call('PostController@index', ['Not used :-(']); | 
用法十二:Contextual Bindings (上下文绑定)
有时候你想在不同的地方给接口不同的实现。这里有 Laravel 文档 里的一个例子:
| 1 | $container | 
现在 PhotoController 和 VideoController 都依赖了 Filesystem 接口,但是收到了不同的实例。
可以像 bind() 那样,给 give() 传闭包:
| 1 | ->when(VideoController::class) | 
或者短名称:
| 1 | $container->instance('s3', $s3Filesystem); | 
用法十三:Binding Parameters to Primitives (绑定初始数据)
当有一个类不仅需要接受一个注入类,还需要注入一个基本值(比如整数)。
还可以通过将变量名称 (而不是接口) 传递给 needs() 并将值传递给 give() 来注入需要的任何值 (字符串、整数等) :
| 1 | $container | 
还可以使用闭包实现延时加载,只在需要的时候取回这个 值 。
| 1 | $container | 
这种情况下,不能传递类或命名的依赖关系(例如,give(‘database.user’)),因为它将作为字面值返回。所以需要使用闭包:
| 1 | $container | 
用法十四: Tagging (标记)
Container 可以用来「标记」有关系的绑定:
| 1 | $container->tag(MyPlugin::class, 'plugin'); | 
这样会以数组的形式取回所有「标记」的实例:
| 1 | foreach ($container->tagged('plugin') as $plugin) { | 
tag() 方法的两个参数都可以接受数组:
| 1 | $container->tag([MyPlugin::class, AnotherPlugin::class], 'plugin'); |