java 设计范式.docx

上传人:b****6 文档编号:7306405 上传时间:2023-01-22 格式:DOCX 页数:18 大小:38.13KB
下载 相关 举报
java 设计范式.docx_第1页
第1页 / 共18页
java 设计范式.docx_第2页
第2页 / 共18页
java 设计范式.docx_第3页
第3页 / 共18页
java 设计范式.docx_第4页
第4页 / 共18页
java 设计范式.docx_第5页
第5页 / 共18页
点击查看更多>>
下载资源
资源描述

java 设计范式.docx

《java 设计范式.docx》由会员分享,可在线阅读,更多相关《java 设计范式.docx(18页珍藏版)》请在冰豆网上搜索。

java 设计范式.docx

java设计范式

第16章设计范式

本章要向大家介绍重要但却并不是那么传统的“范式”(Pattern)程序设计方法。

在向面向对象程序设计的演化过程中,或许最重要的一步就是“设计范式”(DesignPattern)的问世。

它在由Gamma,Helm和Johnson编著的《DesignPatterns》一书中被定义成一个“里程碑”(该书由Addison-Wesley于1995年出版,注释①)。

那本书列出了解决这个问题的23种不同的方法。

在本章中,我们准备伴随几个例子揭示出设计范式的基本概念。

这或许能激起您阅读《DesignPattern》一书的欲望。

事实上,那本书现在已成为几乎所有OOP程序员都必备的参考书。

①:

但警告大家:

书中的例子是用C++写的。

本章的后一部分包含了展示设计进化过程的一个例子,首先是比较原始的方案,经过逐渐发展和改进,慢慢成为更符合逻辑、更为恰当的设计。

该程序(仿真垃圾分类)一直都在进化,可将这种进化作为自己设计方案的一个原型——先为特定的问题提出一个适当的方案,再逐步改善,使其成为解决那类问题一种最灵活的方案。

16.1范式的概念

在最开始,可将范式想象成一种特别聪明、能够自我适应的手法,它可以解决特定类型的问题。

也就是说,它类似一些需要全面认识某个问题的人。

在了解了问题的方方面面以后,最后提出一套最通用、最灵活的解决方案。

具体问题或许是以前见到并解决过的。

然而,从前的方案也许并不是最完善的,大家会看到它如何在一个范式里具体表达出来。

尽管我们称之为“设计范式”,但它们实际上并不局限于设计领域。

思考“范式”时,应脱离传统意义上分析、设计以及实施的思考方式。

相反,“范式”是在一个程序里具体表达一套完整的思想,所以它有时可能出现在分析阶段或者高级设计阶段。

这一点是非常有趣的,因为范式具有以代码形式直接实现的形式,所以可能不希望它在低级设计或者具体实施以前显露出来(而且事实上,除非真正进入那些阶段,否则一般意识不到自己需要一个范式来解决问题)。

范式的基本概念亦可看成是程序设计的基本概念:

添加一层新的抽象!

只要我们抽象了某些东西,就相当于隔离了特定的细节。

而且这后面最引人注目的动机就是“将保持不变的东西身上发生的变化孤立出来”。

这样做的另一个原因是一旦发现程序的某部分由于这样或那样的原因可能发生变化,我们一般都想防止那些改变在代码内部繁衍出其他变化。

这样做不仅可以降低代码的维护代价,也更便于我们理解(结果同样是降低开销)。

为设计出功能强大且易于维护的应用项目,通常最困难的部分就是找出我称之为“领头变化”的东西。

这意味着需要找出造成系统改变的最重要的东西,或者换一个角度,找出付出代价最高、开销最大的那一部分。

一旦发现了“领头变化”,就可以为自己定下一个焦点,围绕它展开自己的设计。

所以设计范式的最终目标就是将代码中变化的内容隔离开。

如果从这个角度观察,就会发现本书实际已采用了一些设计范式。

举个例子来说,继承可以想象成一种设计范式(类似一个由编译器实现的)。

在都拥有同样接口(即保持不变的东西)的对象内部,它允许我们表达行为上的差异(即发生变化的东西)。

合成亦可想象成一种范式,因为它允许我们修改——动态或静态——用于实现类的对象,所以也能修改类的运作方式。

在《DesignPatterns》一书中,大家还能看到另一种范式:

“继承器”(即Iterator,Java1.0和1.1不负责任地把它叫作Enumeration,即“枚举”;Java1.2的集合则改回了“继承器”的称呼)。

当我们在集合里遍历,逐个选择不同的元素时,继承器可将集合的实施细节有效地隐藏起来。

利用继承器,可以编写出通用的代码,以便对一个序列里的所有元素采取某种操作,同时不必关心这个序列是如何构建的。

这样一来,我们的通用代码即可伴随任何能产生继承器的集合使用。

16.1.1单子

或许最简单的设计范式就是“单子”(Singleton),它能提供对象的一个(而且只有一个)实例。

单子在Java库中得到了应用,但下面这个例子显得更直接一些:

909-910页程序

创建单子的关键就是防止客户程序员采用除由我们提供的之外的任何一种方式来创建一个对象。

必须将所有构建器都设为private(私有),而且至少要创建一个构建器,以防止编译器帮我们自动同步一个默认构建器(它会自做聪明地创建成为“友好的”——friendly,而非private)。

此时应决定如何创建自己的对象。

在这儿,我们选择了静态创建的方式。

但亦可选择等候客户程序员发出一个创建请求,然后根据他们的要求动态创建。

不管在哪种情况下,对象都应该保存为“私有”属性。

我们通过公用方法提供访问途径。

在这里,getHandle()会产生指向Singleton的一个句柄。

剩下的接口(getValue()和setValue())属于普通的类接口。

Java也允许通过克隆(Clone)方式来创建一个对象。

在这个例子中,将类设为final可禁止克隆的发生。

由于Singleton是从Object直接继承的,所以clone()方法会保持protected(受保护)属性,不能够使用它(强行使用会造成编译期错误)。

然而,假如我们是从一个类结构中继承,那个结构已经过载了clone()方法,使其具有public属性,并实现了Cloneable,那么为了禁止克隆,需要过载clone(),并掷出一个CloneNotSupportedException(不支持克隆违例),就象第12章介绍的那样。

亦可过载clone(),并简单地返回this。

那样做会造成一定的混淆,因为客户程序员可能错误地认为对象尚未克隆,仍然操纵的是原来的那个。

注意我们并不限于只能创建一个对象。

亦可利用该技术创建一个有限的对象池。

但在那种情况下,可能需要解决池内对象的共享问题。

如果不幸真的遇到这个问题,可以自己设计一套方案,实现共享对象的登记与撤消登记。

16.1.2范式分类

《DesignPatterns》一书讨论了23种不同的范式,并依据三个标准分类(所有标准都涉及那些可能发生变化的方面)。

这三个标准是:

(1)创建:

对象的创建方式。

这通常涉及对象创建细节的隔离,这样便不必依赖具体类型的对象,所以在新添一种对象类型时也不必改动代码。

(2)结构:

设计对象,满足特定的项目限制。

这涉及对象与其他对象的连接方式,以保证系统内的改变不会影响到这些连接。

(3)行为:

对程序中特定类型的行动进行操纵的对象。

这要求我们将希望采取的操作封装起来,比如解释一种语言、实现一个请求、在一个序列中遍历(就象在继承器中那样)或者实现一种算法。

本章提供了“观察器”(Observer)和“访问器”(Visitor)的范式的例子。

《DesignPatterns》为所有这23种范式都分别使用了一节,随附的还有大量示例,但大多是用C++编写的,少数用Smalltalk编写(如看过这本书,就知道这实际并不是个大问题,因为很容易即可将基本概念从两种语言翻译到Java里)。

现在这本书并不打算重复《DesignPatterns》介绍的所有范式,因为那是一本独立的书,大家应该单独阅读。

相反,本章只准备给出一些例子,让大家先对范式有个大致的印象,并理解它们的重要性到底在哪里。

16.2观察器范式

观察器(Observer)范式解决的是一个相当普通的问题:

由于某些对象的状态发生了改变,所以一组对象都需要更新,那么该如何解决?

在Smalltalk的MVC(模型-视图-控制器)的“模型-视图”部分中,或在几乎等价的“文档-视图结构”中,大家可以看到这个问题。

现在我们有一些数据(“文档”)以及多个视图,假定为一张图(Plot)和一个文本视图。

若改变了数据,两个视图必须知道对自己进行更新,而那正是“观察器”要负责的工作。

这是一种十分常见的问题,它的解决方案已包括进标准的java.util库中。

在Java中,有两种类型的对象用来实现观察器范式。

其中,Observable类用于跟踪那些当发生一个改变时希望收到通知的所有个体——无论“状态”是否改变。

如果有人说“好了,所有人都要检查自己,并可能要进行更新”,那么Observable类会执行这个任务——为列表中的每个“人”都调用notifyObservers()方法。

notifyObservers()方法属于基础类Observable的一部分。

在观察器范式中,实际有两个方面可能发生变化:

观察对象的数量以及更新的方式。

也就是说,观察器范式允许我们同时修改这两个方面,不会干扰围绕在它周围的其他代码。

下面这个例子类似于第14章的ColorBoxes示例。

箱子(Boxes)置于一个屏幕网格中,每个都初始化一种随机的颜色。

此外,每个箱子都“实现”(implement)了“观察器”(Observer)接口,而且随一个Observable对象进行了注册。

若点击一个箱子,其他所有箱子都会收到一个通知,指出一个改变已经发生。

这是由于Observable对象会自动调用每个Observer对象的update()方法。

在这个方法内,箱子会检查被点中的那个箱子是否与自己紧邻。

若答案是肯定的,那么也修改自己的颜色,保持与点中那个箱子的协调。

912-914页程序

如果是首次查阅Observable的联机帮助文档,可能会多少感到有些困惑,因为它似乎表明可以用一个原始的Observable对象来管理更新。

但这种说法是不成立的;大家可自己试试——在BoxObserver中,创建一个Observable对象,替换BoxObservable对象,看看会有什么事情发生。

事实上,什么事情也不会发生。

为真正产生效果,必须从Observable继承,并在衍生类代码的某个地方调用setChanged()。

这个方法需要设置“changed”(已改变)标志,它意味着当我们调用notifyObservers()的时候,所有观察器事实上都会收到通知。

在上面的例子中,setChanged()只是简单地在notifyObservers()中调用,大家可依据符合实际情况的任何标准决定何时调用setChanged()。

BoxObserver包含了单个Observable对象,名为notifier。

每次创建一个OCBox对象时,它都会同notifier联系到一起。

在OCBox中,只要点击鼠标,就会发出对notifyObservers()方法的调用,并将被点中的那个对象作为一个参数传递进去,使收到消息(用它们的update()方法)的所有箱子都能知道谁被点中了,并据此判断自己是否也要变动。

通过notifyObservers()和update()中的代码的结合,我们可以应付一些非常复杂的局面。

在notifyObservers()方法中,表面上似乎观察器收到通知的方式必须在编译期间固定下来。

然而,只要稍微仔细研究一下上面的代码,就会发现BoxObserver或OCBox中唯一需要留意是否使用BoxObservable的地方就是创建Observable对象的时候——从那时开始,所有东西都会使用基本的Observable接口。

这意味着以后若想更改通知方式,可以继承其他Observable类,并在运行期间交换它们。

16.3模拟垃圾回收站

这个问题的本质是若将垃圾丢进单个垃圾筒,事实上是未经分类的。

但在以后,某些特殊的信息必须恢复,以便对垃圾正确地归类。

在最开始的解决方案中,RTTI扮演了关键的角色(详见第11章)。

这并不是一种普通的设计,因为它增加了一个新的限制。

正是这个限制使问题变得非常有趣——它更象我们在工作中碰到的那些非常麻烦的问题。

这个额外的限制是:

垃圾抵达垃圾回收站时,它们全都是混合在一起的。

程序必须为那些垃圾的分类定出一个模型。

这正是RTTI发挥作用的地方:

我们有大量不知名的垃圾,程序将正确判断出它们所属的类型。

915-917页程序

要注意的第一个地方是package语句:

packagec16.recyclea;

这意味着在本书采用的源码目录中,这个文件会被置入从c16(代表第16章的程序)分支出来的recyclea子目录中。

第17章的解包工具会负责将其置入正确的子目录。

之所以要这样做,是因为本章会多次改写这个特定的例子;它的每个版本都会置入自己的“包”(package)内,避免类名的冲突。

其中创建了几个Vector对象,用于容纳Trash句柄。

当然,Vector实际容纳的是Object(对象),所以它们最终能够容纳任何东西。

之所以要它们容纳Trash(或者从Trash衍生出来的其他东西),唯一的理由是我们需要谨慎地避免放入除Trash以外的其他任何东西。

如果真的把某些“错误”的东西置入Vector,那么不会在编译期得到出错或警告提示——只能通过运行期的一个违例知道自己已经犯了错误。

Trash句柄加入后,它们会丢失自己的特定标识信息,只会成为简单的Object句柄(上溯造型)。

然而,由于存在多形性的因素,所以在我们通过Enumerationsorter调用动态绑定方法时,一旦结果Object已经造型回Trash,仍然会发生正确的行为。

sumValue()也用一个Enumeration对Vector中的每个对象进行操作。

表面上持,先把Trash的类型上溯造型到一个集合容纳基础类型的句柄,再回过头重新下溯造型,这似乎是一种非常愚蠢的做法。

为什么不只是一开始就将垃圾置入适当的容器里呢?

(事实上,这正是拨开“回收”一团迷雾的关键)。

在这个程序中,我们很容易就可以换成这种做法,但在某些情况下,系统的结构及灵活性都能从下溯造型中得到极大的好处。

该程序已满足了设计的初衷:

它能够正常工作!

只要这是个一次性的方案,就会显得非常出色。

但是,真正有用的程序应该能够在任何时候解决问题。

所以必须问自己这样一个问题:

“如果情况发生了变化,它还能工作吗?

”举个例子来说,厚纸板现在是一种非常有价值的可回收物品,那么如何把它集成到系统中呢(特别是程序很大很复杂的时候)?

由于前面在switch语句中的类型检查编码可能散布于整个程序,所以每次加入一种新类型时,都必须找到所有那些编码。

若不慎遗漏一个,编译器除了指出存在一个错误之外,不能再提供任何有价值的帮助。

RTTI在这里使用不当的关键是“每种类型都进行了测试”。

如果由于类型的子集需要特殊的对待,所以只寻找那个子集,那么情况就会变得好一些。

但假如在一个switch语句中查找每一种类型,那么很可能错过一个重点,使最终的代码很难维护。

在下一节中,大家会学习如何逐步对这个程序进行改进,使其显得越来越灵活。

这是在程序设计中一种非常有意义的例子。

16.4改进设计

《DesignPatterns》书内所有方案的组织都围绕“程序进化时会发生什么变化”这个问题展开。

对于任何设计来说,这都可能是最重要的一个问题。

若根据对这个问题的回答来构造自己的系统,就可以得到两个方面的结果:

系统不仅更易维护(而且更廉价),而且能产生一些能够重复使用的对象,进而使其他相关系统的构造也变得更廉价。

这正是面向对象程序设计的优势所在,但这一优势并不是自动体现出来的。

它要求对我们对需要解决的问题有全面而且深入的理解。

在这一节中,我们准备在系统的逐步改进过程中向大家展示如何做到这一点。

就目前这个回收系统来说,对“什么会变化”这个问题的回答是非常普通的:

更多的类型会加入系统。

因此,设计的目标就是尽可能简化这种类型的添加。

在回收程序中,我们准备把涉及特定类型信息的所有地方都封装起来。

这样一来(如果没有别的原因),所有变化对那些封装来说都是在本地进行的。

这种处理方式也使代码剩余的部分显得特别清爽。

16.4.1“制作更多的对象”

这样便引出了面向对象程序设计时一条常规的准则,我最早是在GradyBooch那里听说的:

“若设计过于复杂,就制作更多的对象”。

尽管听起来有些暧昧,且简单得可笑,但这确实是我知道的最有用一条准则(大家以后会注意到“制作更多的对象”经常等同于“添加另一个层次的迂回”)。

一般情况下,如果发现一个地方充斥着大量繁复的代码,就需要考虑什么类能使它显得清爽一些。

用这种方式整理系统,往往会得到一个更好的结构,也使程序更加灵活。

首先考虑Trash对象首次创建的地方,这是main()里的一个switch语句:

919-920页程序

这些代码显然“过于复杂”,也是新类型加入时必须改动代码的场所之一。

如果经常都要加入新类型,那么更好的方案就是建立一个独立的方法,用它获取所有必需的信息,并创建一个句柄,指向正确类型的一个对象——已经上溯造型到一个Trash对象。

在《DesignPatterns》中,它被粗略地称呼为“创建范式”。

要在这里应用的特殊范式是Factory方法的一种变体。

在这里,Factory方法属于Trash的一名static(静态)成员。

但更常见的一种情况是:

它属于衍生类中一个被过载的方法。

Factory方法的基本原理是我们将创建对象所需的基本信息传递给它,然后返回并等候句柄(已经上溯造型至基础类型)作为返回值出现。

从这时开始,就可以按多形性的方式对待对象了。

因此,我们根本没必要知道所创建对象的准确类型是什么。

事实上,Factory方法会把自己隐藏起来,我们是看不见它的。

这样做可防止不慎的误用。

如果想在没有多形性的前提下使用对象,必须明确地使用RTTI和指定造型。

但仍然存在一个小问题,特别是在基础类中使用更复杂的方法(不是在这里展示的那种),且在衍生类里过载(覆盖)了它的前提下。

如果在衍生类里请求的信息要求更多或者不同的参数,那么该怎么办呢?

“创建更多的对象”解决了这个问题。

为实现Factory方法,Trash类使用了一个新的方法,名为factory。

为了将创建数据隐藏起来,我们用一个名为Info的新类包含factory方法创建适当的Trash对象时需要的全部信息。

下面是Info一种简单的实现方式:

920-921页程序

Info对象唯一的任务就是容纳用于factory()方法的信息。

现在,假如出现了一种特殊情况,factory()需要更多或者不同的信息来新建一种类型的Trash对象,那么再也不需要改动factory()了。

通过添加新的数据和构建器,我们可以修改Info类,或者采用子类处理更典型的面向对象形式。

用于这个简单示例的factory()方法如下:

921页上程序

在这里,对象的准确类型很容易即可判断出来。

但我们可以设想一些更复杂的情况,factory()将采用一种复杂的算法。

无论如何,现在的关键是它已隐藏到某个地方,而且我们在添加新类型时知道去那个地方。

新对象在main()中的创建现在变得非常简单和清爽:

921页下程序

我们在这里创建了一个Info对象,用于将数据传入factory();后者在内存堆中创建某种Trash对象,并返回添加到Vectorbin内的句柄。

当然,如果改变了参数的数量及类型,仍然需要修改这个语句。

但假如Info对象的创建是自动进行的,也可以避免那个麻烦。

例如,可将参数的一个Vector传递到Info对象的构建器中(或直接传入一个factory()调用)。

这要求在运行期间对参数(自变量)进行分析与检查,但确实提供了非常高的灵活程度。

大家从这个代码可看出Factory要负责解决的“领头变化”问题:

如果向系统添加了新类型(发生了变化),唯一需要修改的代码在Factory内部,所以Factory将那种变化的影响隔离出来了。

16.4.2用于原型创建的一个范式

上述设计方案的一个问题是仍然需要一个中心场所,必须在那里知道所有类型的对象:

在factory()方法内部。

如果经常都要向系统添加新类型,factory()方法为每种新类型都要修改一遍。

若确实对这个问题感到苦恼,可试试再深入一步,将与类型有关的所有信息——包括它的创建过程——都移入代表那种类型的类内部。

这样一来,每次新添一种类型的时候,需要做的唯一事情就是从一个类继承。

为将涉及类型创建的信息移入特定类型的Trash里,必须使用“原型”(prototype)范式(来自《DesignPatterns》那本书)。

这里最基本的想法是我们有一个主控对象序列,为自己感兴趣的每种类型都制作一个。

这个序列中的对象只能用于新对象的创建,采用的操作类似内建到Java根类Object内部的clone()机制。

在这种情况下,我们将克隆方法命名为tClone()。

准备创建一个新对象时,要事先收集好某种形式的信息,用它建立我们希望的对象类型。

然后在主控序列中遍历,将手上的信息与主控序列中原型对象内任何适当的信息作对比。

若找到一个符合自己需要的,就克隆它。

采用这种方案,我们不必用硬编码的方式植入任何创建信息。

每个对象都知道如何揭示出适当的信息,以及如何对自身进行克隆。

所以一种新类型加入系统的时候,factory()方法不需要任何改变。

为解决原型的创建问题,一个方法是添加大量方法,用它们支持新对象的创建。

但在Java1.1中,如果拥有指向Class对象的一个句柄,那么它已经提供了对创建新对象的支持。

利用Java1.1的“反射”(已在第11章介绍)技术,即便我们只有指向Class对象的一个句柄,亦可正常地调用一个构建器。

这对原型问题的解决无疑是个完美的方案。

原型列表将由指向所有想创建的Class对象的一个句柄列表间接地表示。

除此之外,假如原型处理失败,则factory()方法会认为由于一个特定的Class对象不在列表中,所以会尝试装载它。

通过以这种方式动态装载原型,Trash类根本不需要知道自己要操纵的是什么类型。

因此,在我们添加新类型时不需要作出任何形式的修改。

于是,我们可在本章剩余的部分方便地重复利用它。

923-925页程序

基本Trash类和sumValue()还是象往常一样。

这个类剩下的部分支持原型范式。

大家首先会看到两个内部类(被设为static属性,使其成为只为代码组织目的而存在的内部类),它们描述了可能出现的违例。

在它后面跟随的是一个VectortrashTypes,用于容纳Class句柄。

在Trash.factory()中,Info对象id(Info类的另一个版本,与前面讨论的不同)内部的String包含了要创建的那种Trash的类型名称。

这个String会与列表中的Class名比较。

若存在相符的,那便是要创建的对象。

当然,还有很多方法可以决定我们想创建的对象。

之所以要采用这种方法,是因为从一个文件读入的信息可以转换成对象。

发现自己要创建的Trash(垃圾)种类后,接下来就轮到“反射”方法大显身手了。

getConstructor()方法需要取得自己的参数——由Class句柄构成的一个数组。

这个数组代表着不同的参数,并按它们正确的顺序排列,以便我们查找的构建器使用。

在这儿,该数组是用Java1.1的数组创建语法动态创建的:

newClass[]{double.class}

这个代码假定所有Trash类型都有一个需要double数值的构建器(注意double.class与Double.class是不同的)。

若考虑一种更灵活的方案,亦可调用getConstructors(),令其返回可用构建器的一个数组。

从getConstructors()返回的是指向一个Constructor对象的句柄(该对象是java.lang.reflect的一部分)。

我们用方法newInstance()动态地调用构建器。

该方法需要获取包含了实际参数的一个Object数组。

这个数组同样是按Java1.1的语法创建的:

newObject[]{newDouble(info.data)}

在这种情况下,double必须置入一个封装(容器)类的内部,使其真正成为这个对象数组的一部分。

通过调用newInstance(),会提取出double,但大家可能会觉得稍微有些迷惑——参数既可能是double,也可能是Double,但在调用的时候必须用Doub

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 小学教育 > 语文

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1