欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  后端开发

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

程序员文章站 2022-06-14 13:18:54
...
基于接口的编程

作为一个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类的定义

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类的定义

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类的定义

command = $command;
    }

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

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

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

例子 2-11 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教程有兴趣的朋友有所帮助。