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 相关文章推荐
PHP的FTP学习(四)
Oct 09 PHP
IP138 IP地址查询小偷实现代码
Feb 15 PHP
php 中英文语言转换类
Sep 07 PHP
PHP执行Curl时报错提示CURL ERROR: Recv failure: Connection reset by peer的解决方法
Jun 26 PHP
php计算指定目录下文件占用空间的方法
Mar 13 PHP
推荐十款免费 WordPress 插件
Mar 24 PHP
php curl模拟post请求和提交多维数组的示例代码
Nov 19 PHP
PHP 数组基本操作小结(推荐)
Jun 13 PHP
Yii2验证器(Validator)用法分析
Jul 23 PHP
Apache PHP MySql安装配置图文教程
Aug 27 PHP
PHP函数按引用传递参数及函数可选参数用法示例
Jun 04 PHP
基于thinkphp6.0的success、error实现方法
Nov 05 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
php split汉字
2009/06/05 PHP
PHP原生模板引擎 最简单的模板引擎
2012/04/25 PHP
比较简单的百度网盘文件直链PHP代码
2013/03/24 PHP
PHP中static关键字以及与self关键字的区别
2015/07/01 PHP
基础的WordPress插件制作教程
2015/11/24 PHP
FleaPHP框架数据库查询条件($conditions)写法总结
2016/03/19 PHP
PHP实现简单实用的分页类代码
2016/04/08 PHP
laravel 事件/监听器实例代码
2019/04/12 PHP
JavaScript入门教程(1) 什么是JS
2009/01/31 Javascript
jQuery基本过滤选择器使用介绍
2013/04/18 Javascript
JQuery Highcharts 动态生成图表的方法
2013/11/15 Javascript
js日期对象兼容性的处理方法
2014/01/28 Javascript
基于JQuery实现分隔条的功能
2016/06/17 Javascript
Bootstrap3使用typeahead插件实现自动补全功能
2016/07/07 Javascript
jQuery的deferred对象使用详解
2016/09/25 Javascript
Bootstrap整体框架之CSS12栅格系统
2016/12/15 Javascript
微信小程序实现瀑布流布局与无限加载的方法详解
2017/05/12 Javascript
jQuery动画_动力节点节点Java学院整理
2017/07/04 jQuery
JS实现简单的浮动碰撞效果示例
2017/12/28 Javascript
puppeteer库入门初探
2019/01/09 Javascript
基于Bootstrap和JQuery实现动态打开和关闭tab页的实例代码
2019/06/10 jQuery
Vue通过for循环随机生成不同的颜色或随机数的实例
2019/11/09 Javascript
Vue项目中数据的深度监听或对象属性的监听实例
2020/07/17 Javascript
菜鸟使用python实现正则检测密码合法性
2016/01/05 Python
python开发之list操作实例分析
2016/02/22 Python
python编程实现随机生成多个椭圆实例代码
2018/01/03 Python
Python统计单词出现的次数
2018/04/04 Python
对python GUI实现完美进度条的示例详解
2018/12/13 Python
使用Python制作新型冠状病毒实时疫情图
2020/01/28 Python
python如何通过twisted搭建socket服务
2020/02/03 Python
python 如何调用 dubbo 接口
2020/09/24 Python
HTML5 Canvas的性能提高技巧经验分享
2013/07/02 HTML / CSS
用canvas显示验证码的实现
2020/04/10 HTML / CSS
Paradox London官方网站:英国新娘鞋婚礼鞋品牌
2019/08/29 全球购物
高中的职业生涯规划书
2013/12/28 职场文书
Java数组与堆栈相关知识总结
2021/06/29 Java/Android