首页 >后端开发 >php教程 >[Modern PHP] 第二章 新特性之二 基于接口的编程

[Modern PHP] 第二章 新特性之二 基于接口的编程

WBOY
WBOY原创
2016-07-30 13:31:19976浏览

基于接口的编程

作为一个PHP程序员,学习如何基于接口进行编程改变了我的人生,大大提升了我通过集成第三方PHP组件来完善我的项目的能力。接口并不是新的功能,但是它却是你日常工作中必须了解和使用的重要特性。

那么PHP的接口到底是什么?接口是两个PHP对象之间的契约,一个对象调用另一个对象时并不需要知道对方是什么,而只需要知道对方能做什么。接口可以降低我们代码依赖关系的耦合性,允许我们的代码调用任何实现了期望接口的第三方代码。我们只需要关心第三方的代码是否实现了接口就行,而根本不用关心第三方代码是如何实现这些接口的。我们来看一个现实中的例子。

假设我到佛罗里达的迈阿密参加Sunshine PHP开发者大会。我想去城里逛逛,所以我直接去了当地的汽车租赁公司。他们有一辆现代紧凑型的小车,一辆斯巴鲁的旅行车和一辆布加迪威龙(太让我惊讶了)。我知道我只是想借助某个交通工具去城里逛逛,这三辆车都可以满足我的需求。但是每辆车又是那么的与众不同。现代雅绅特还不错啦,但是我喜欢更有活力一点的。我没有孩子,所以旅行车还是大了点。那么,好吧就选布加迪好了。

现实是,我驾驶这三辆车中的任意一辆都可以,因为它们都共享了公用的、已知的接口。每辆车都有一个方向盘、一个油门踏板、一个刹车踏板和转向灯,每辆车都使用汽油作为燃料。但是布加迪的动力强大到我都无法驾驭,但是现代车的驾驶接口却和它一模一样。因为三辆车都共享了同样的已知的接口,但我有机会可以选择我更喜欢的车型(实话实话,最终我可能还是会选择现代)。

在PHP的面向对象方面也有同样的概念。如果我的代码中使用了某个特定的类(代表了特定的实现)的对象,那么我的代码的功能也固然变得非常有限,因为它只能永远使用那个类的对象。然而,如果我的代码将要用到的是一个接口,我的代码立刻就能知道如何去使用任何实现了这个接口的对象。我的代码根本不用关心接口是如何实现的,我的代码只关心对象是否实现了接口,让我们用一个例子把这一切解释清楚。

假设我们有一个PHP类名叫DocumentStore可以从不同的数据源中获取数据,它可以从一个远程的地址中获取HTML,也可以从文档流中读取数据,还可以获取终端的命令行输出。每个保存在DocumentStore实例中的文档都有一个唯一的ID。例子 2-6 展示了DocumentStore类。

例子 2-6 DocumentStore类的定义

class DocumentStore
{
    protected $data = [];

    public function addDocument(Documentable $document)
    {
        $key = $document->getId();
        $value = $document->getContent();
        $this->data[$key] = $value;
    }

    public function getDocuments()
    {
        return $this->data;
    }
}

如果addDocument方法只接收Documentable类的实例作为参数怎么才能实现前面描述的功能?很好的观察力。实际上Documentable不是一个类,而是一个接口,看看例子 2-7里接口的定义

例子 2-7 Documentable接口的定义

interface Documentable
{
    public function getId();
    public function getContent();
}

在接口的定义中我们能看到任何实现了Documentable接口的对象都必须定义一个public的getId()方法和一个public的getContent()方法。

那么这样做的好处是什么?好处是我们可以分别构造多个功能迥异的文档获取类。例子 2-8展示了一个通过curl从远程地址获取HTML的接口实现。

例子 2-8 HtmlDocument类的定义

<?php class HtmlDocument implements Documentable
{
    protected $url;

    public function __construct($url)
    {
        $this->url = $url;
    }

    public function getId()
    {
        return $this->url;
    }

    public function getContent()
    {
        $ch = curl_init();
        curl_setopt($ch, CURLOPT_URL, $this->url);
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
        curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 3);
        curl_setopt($ch, CURLOPT_FOLLOWLOCATION, 1);
        curl_setopt($ch, CURLOPT_MAXREDIRS, 3);
        $html = curl_exec($ch);
        curl_close($ch);

        return $html;
    }
}

另一个实现(例子 2-9)可以读取给定资源中的数据流。

例子 2-9 StreamDocument类的定义

<?php class StreamDocument implements Documentable
{
    protected $resource;
    protected $buffer;

    public function __construct($resource, $buffer = 4096)
    {
        $this->resource = $resource;
        $this->buffer = $buffer;
    }

    public function getId()
    {
        return 'resource-' . (int)$this->resource;
    }

    public function getContent()
    {
        $streamContent = '';
        rewind($this->resource);
        while (feof($this->resource) === false) {
            $streamContent .= fread($this->resource, $this->buffer);
        }

        return $streamContent;
    }
}

最后一个实现(例子 2-10)可以执行一个终端的命令并获取执行结果

例子 2-10 CommandOutputDocument类的定义

<?php class CommandOutputDocument implements Documentable
{
    protected $command;

    public function __construct($command)
    {
        $this->command = $command;
    }

    public function getId()
    {
        return $this->command;
    }

    public function getContent()
    {
        return shell_exec($this->command);
    }
}

例子 2-11 展示了我们如何使用DocumentStore类来操作我们实现了Documentable接口的三个文档收集类

例子 2-11 DocumentStore

<?php $documentStore = new DocumentStore();

// Add HTML document
$htmlDoc = new HtmlDocument(&#39;http://php.net&#39;);
$documentStore->addDocument($htmlDoc);

// Add stream document
$streamDoc = new StreamDocument(fopen('stream.txt', 'rb'));
$documentStore->addDocument($streamDoc);

// Add terminal command document
$cmdDoc = new CommandOutputDocument('cat /etc/hosts');
$documentStore->addDocument($cmdDoc);

print_r($documentStore->getDocuments());

这样做最大的亮点在于HtmlDocument、StreamDocument和CommandOutputDocument类除了接口一致外其它方面完全不同。

时至今日,对接口进行编程创造出了更灵活的代码,我们不再关心具体的实现,而是将这些工作交给其他人来完成。越来越多的人(譬如你的同事,你的开源项目的用户,或者你从来都没有见过面的开发者)可以写出与你能够彼此无缝对接的代码,而他们仅仅只需要了解接口即可。

以上就介绍了[Modern PHP] 第二章 新特性之二 基于接口的编程,包括了方面的内容,希望对PHP教程有兴趣的朋友有所帮助。

声明:
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn