基于接口的编程
作为一个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; } }
例子 2-7 Documentable接口的定义
interface Documentable { public function getId(); public function 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 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 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
<?php $documentStore = new DocumentStore(); // Add HTML document $htmlDoc = new HtmlDocument('http://php.net'); $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());
时至今日,对接口进行编程创造出了更灵活的代码,我们不再关心具体的实现,而是将这些工作交给其他人来完成。越来越多的人(譬如你的同事,你的开源项目的用户,或者你从来都没有见过面的开发者)可以写出与你能够彼此无缝对接的代码,而他们仅仅只需要了解接口即可。
以上就介绍了[Modern PHP] 第二章 新特性之二 基于接口的编程,包括了方面的内容,希望对PHP教程有兴趣的朋友有所帮助。