Laravel5中contracts详解


Posted in PHP onMarch 02, 2015

我们先来看看官方文档中对contracts的定义:

Laravel's Contracts are a set of interfaces that define the core services provided by the framework.
意思是说Laravel的Contracts是一个由 框架提供 的定义了 核心服务接口 的集合。

也就是说,每一个Contract都是一个接口,对应一个框架核心服务。

那它的意义何在?官网给出的解释也很简单:使用接口是为了 松耦合 和 简单 。

先不讲大道理,先来点干货,看看怎么使用contract

先浏览下contracts接口列表:

Illuminate\Contracts\Auth\Guard

Illuminate\Contracts\Auth\PasswordBroker

Illuminate\Contracts\Bus\Dispatcher

Illuminate\Contracts\Cache\Repository

Illuminate\Contracts\Cache\Factory

Illuminate\Contracts\Config\Repository

Illuminate\Contracts\Container\Container

Illuminate\Contracts\Cookie\Factory

Illuminate\Contracts\Cookie\QueueingFactory

Illuminate\Contracts\Encryption\Encrypter

Illuminate\Contracts\Routing\Registrar

…… 太多了,懒得继续贴了,官网手册里有。我们就拿 Illuminate\Contracts\Routing\Registrar 这个contract来演示一下吧。
首先,打开 app/Providers/AppServiceProvider.php,注意register方法:

public function register()

{

    $this->app->bind(

        'Illuminate\Contracts\Auth\Registrar',

        'App\Services\Registrar'

    );

}

$this->app 就是Application对象,也是容器对象,通过 $this->app->bind 方法我们绑定了一个实现Illuminate\Contracts\Auth\Registrar接口的类App\Services\Registrar。

注意,Illuminate\Contracts\Auth\Registrar就是一个contract。App\Services\Registrar 这个类文件在 app/Services/Registrar.php。

接着我们看 App\Http\Controllers\Auth\AuthController 这个控制器类,看到它有 __construct 构造函数:

public function __construct(Guard $auth, Registrar $registrar)

{

    $this->auth = $auth;

    $this->registrar = $registrar;
    $this->middleware('guest', ['except' => 'getLogout']);

}

它有两个参数,对应的类命名空间在脚本开头可以看到:

use Illuminate\Contracts\Auth\Guard;

use Illuminate\Contracts\Auth\Registrar;

这两个都是contract,但我们这里就拿 Registrar 说,我们注意到这里面只是通过参数类型指明了$registrar的接口类型,而实际调用的时候实际上是 App\Services\Registrar 这个类,这就是依赖注入的特性了,Laravel会自动在容器中搜索实现了接口Illuminate\Contracts\Auth\Registrar的类或对象,有的话就取出来作为实际参数传到构造函数里。

整个使用流程其实就可以总结为两个步骤:

向容器中注册实现contract接口的对象。
构造函数参数类型指定为contract接口类,框架会自动找到符合条件的对象。
那么再来说说contract的好处。

松耦合

官网给了一个例子解释什么是紧耦合以及Contract接口为何能够松耦合。

先来看看紧耦合的代码:

<?php namespace App\Orders;

class Repository {

    /**

     * The cache.

     */

    protected $cache;

    /**

     * Create a new repository instance.

     *

     * @param  \SomePackage\Cache\Memcached  $cache

     * @return void

     */

    public function __construct(\SomePackage\Cache\Memcached $cache)

    {

        $this->cache = $cache;

    }

    /**

     * Retrieve an Order by ID.

     *

     * @param  int  $id

     * @return Order

     */

    public function find($id)

    {

        if ($this->cache->has($id))

        {

            //

        }

    }

}

可以看到构造函数中注入了一个详细的缓存实现 \SomePackage\Cache\Memcached ,如果换Redis作为缓存服务器或者更改了api方法,就需要修改,而如果项目很大,你不知道还有多少地方需要修改。

那么,Contract接口是如何解决这个问题的?请看代码:

<?php namespace App\Orders;

use Illuminate\Contracts\Cache\Repository as Cache;

class Repository {

    /**

     * Create a new repository instance.

     *

     * @param  Cache  $cache

     * @return void

     */

    public function __construct(Cache $cache)

    {

        $this->cache = $cache;

    }

}

注意,缓存实现我们使用了一个接口,也就是contract,Illuminate\Contracts\Cache\Repository,因为它只是接口,不需要关心背后是memcache还是redis。

简单性

如果所有服务都使用接口定义,就可以很简单的决定一个服务需要的功能,更加容易维护和扩展,并且contract接口还能看作一个简洁的文档便于阅读。

PHP 相关文章推荐
别人整理的服务器变量:$_SERVER
Oct 20 PHP
php 删除一个数组中的某个值.兼容多维数组!
Feb 18 PHP
注册页面之前先验证用户名是否存在的php代码
Jul 14 PHP
使用PHP求两个文件的相对路径
Jun 20 PHP
thinkphp实现多语言功能(语言包)
Mar 04 PHP
去掉destoon资讯内容页keywords关键字自带的文章标题的方法
Aug 21 PHP
Yii不依赖Model的表单生成器用法实例
Dec 04 PHP
从性能方面考虑PHP下载远程文件的3种方法
Dec 29 PHP
php微信公众平台开发(三)订阅事件处理
Dec 06 PHP
php使用PDO执行SQL语句的方法分析
Feb 16 PHP
php实现的错误处理封装类实例
Jun 20 PHP
php layui实现前端多图上传实例
Jul 30 PHP
php打印一个边长为N的实心和空心菱型的方法
Mar 02 #PHP
初识laravel5
Mar 02 #PHP
避免Smarty与CSS语法冲突的方法
Mar 02 #PHP
实现PHP+Mysql无限分类的方法汇总
Mar 02 #PHP
Java和PHP在Web开发方面对比分析
Mar 01 #PHP
php中return的用法实例分析
Feb 28 #PHP
php多次include后导致全局变量global失效的解决方法
Feb 28 #PHP
You might like
用cookies来跟踪识别用户
2006/10/09 PHP
vBulletin HACK----显示话题大小和打开新窗口于论坛索引页
2006/10/09 PHP
php str_pad() 将字符串填充成指定长度的字符串
2010/02/23 PHP
PHP执行linux系统命令的常用函数使用说明
2010/04/27 PHP
php获取本周开始日期和结束日期的方法
2015/03/09 PHP
解决laravel5.4下的group by报错的问题
2019/10/16 PHP
php 下 html5 XHR2 + FormData + File API 上传文件操作实例分析
2020/02/28 PHP
PHP实现创建一个RPC服务操作示例
2020/02/23 PHP
URL编码转换,escape() encodeURI() encodeURIComponent()
2006/12/27 Javascript
jQuery EasyUI API 中文文档 - Documentation 文档
2011/09/29 Javascript
Javascript this 的一些学习总结
2012/08/02 Javascript
js时间日期和毫秒的相互转换
2013/02/22 Javascript
node.js中的fs.chmodSync方法使用说明
2014/12/18 Javascript
javascript 用函数实现继承详解
2016/05/28 Javascript
jQuery插件实现图片轮播特效
2016/06/16 Javascript
微信小程序 点击控件后选中其它反选实例详解
2017/02/21 Javascript
vue项目中axios请求网络接口封装的示例代码
2018/12/18 Javascript
JS+HTML5本地存储Localstorage实现注册登录及验证功能示例
2020/02/10 Javascript
python之文件的读写和文件目录以及文件夹的操作实现代码
2016/08/28 Python
Python 通过pip安装Django详细介绍
2017/04/28 Python
对python中array.sum(axis=?)的用法介绍
2018/06/28 Python
Python合并同一个文件夹下所有PDF文件的方法
2019/03/11 Python
如何使用Python实现斐波那契数列
2019/07/02 Python
NumPy中的维度Axis详解
2019/11/26 Python
安装PyInstaller失败问题解决
2019/12/14 Python
Python 实现自动登录+点击+滑动验证功能
2020/06/10 Python
python实现npy格式文件转换为txt文件操作
2020/07/01 Python
Python使用eval函数执行动态标表达式过程详解
2020/10/17 Python
canvas之自定义头像功能实现代码示例
2017/09/29 HTML / CSS
lookfantastic荷兰:在线购买奢华护肤、护发和化妆品
2018/11/27 全球购物
九一八事变纪念日演讲稿
2014/09/14 职场文书
领导干部群众路线个人对照检查材料思想汇报
2014/09/30 职场文书
清明节扫墓活动总结
2015/02/09 职场文书
初中政治教学工作总结
2015/08/13 职场文书
redis cluster支持pipeline的实现思路
2021/06/23 Redis
经典《舰娘》游改全新动画预告 预定11月开播
2022/04/01 日漫