前言
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'); |