日期:2010-09-14  浏览次数:20486 次

Aiyiweb.Com提示:接口的改变,是一个需求程序员们必须(虽然很不情愿)接受和处理的普遍问题。程序提供者们修正他们的代码;系统库被修正;各种程序文语以及相关库的发展和进化。我孩子的无数玩具中有一个简要地描述了这个两难局面:你无法合理安排一个不得其所的人。

上文:《PHP设计模式引见》第十二章 装饰器模式

《PHP设计模式引见》第十三章 适配器模式

接口的改变,是一个需求程序员们必须(虽然很不情愿)接受和处理的普遍问题。程序提供者们修正他们的代码;系统库被修正;各种程序文语以及相关库的发展和进化。我孩子的无数玩具中有一个简要地描述了这个两难局面:你无法合理安排一个不得其所的人。

问题

你如何避免因外部库的API改变而带来的不便?假如你写了一个库,你能否提供一种方法允许你软件的现有用户进行完满地升级,即便你曾经改变了你的API?为了更好地适宜于你的需求,你应该如何改变一个对象的接口?

处理方案

适配器(Adapter)模式为对象提供了一种完全不同的接口。你可以运用适配器(Adapter)来实现一个不同的类的常见接口,同时避免了因升级和拆解客户代码所惹起的纠纷。

考虑一下当(不是假设!)一个第三方库的API改变将会发生什么。过去你只能是咬紧牙关修正所有的客户代码,而情况往往还不那么简单。你可能正从事一项新的项目,它要用到新版本的库所带来的特性,但你曾经拥有许多旧的使用程序,并且它们与以前旧版本的库交互运转地很好。你将无法证明这些新特性的利用价值,如果这次升级意味着将要涉及到其它使用程序的客户代码。

注:控制体模式
适配器(Adapter)模式是控制体模式的最新范例。一个适配器(Adapter)的结构类似于代理服务器(Proxy)和修饰器(Decorator),而它们的不同之处在于,适配器(Adapter)的目的是改变封装类的接口,代理服务器(Proxy)和修饰器(Decorator)则是保持接口不变。

样本代码

让我们看看当API改变时,如何保护使用程序不受影响。

假设你费尽心思寻觅合适的库,最后终于找到了HwLib,一个(假设的)被设计用来发送信息的代码集。

下面是HwLib类的源代码:

// PHP4
/**
* the HwLib helps programmers everywhere write their first program
* @package HelloWorld
* @version 1
*/
class HwLib {
/**
* Say “Hello”
* @deprec this function is going away in the future
* @return string
*/
function hello() {
return ‘Hello ‘;
}
/**
* target audience
* @return string
*/
function world() {
return ‘World!’;
}
}

下面是库运转的范例:

$hw =& new HwLib;
echo $hw->hello(), $hw->world();

HwLib有完备的说明文档。在文档中作者曾经明确指出hello()方法会在未来的版本中不被支持(甚至淘汰)。
接下来,如今假设第二版的HwLib曾经发布。一个全新的greet()方法代替了hello()。
下面是这个库的新版本(注释已被抽取掉):

// version 2
class HwLib {
function greet() {
return ‘Greetings and Salutations ‘;
}
function world() {
return ‘World!’;
}
}