最大限制地提高代码的可重用性,克服传统面向对象编程方法在可重用性方面的不足

编程|对象

    重用是一种神话,这似乎正在日渐成为编程人员的一种共识。然而,重用可能难以实现,因为传统面向对象编程方法在可重用性方面存在一些不足。本技巧说明了组成支持重用的一种不同方法的三个步骤。 第一步:将功能移出类实例方法由于类继承机制缺乏精确性,因此对于代码重用来说它并不是一种最理想的机制。也就是说,如果您要重用某个类的单个方法,就必须继承该类的其他方法以及数据成员。这种累赘不必要地将要重用此方法的代码复杂化了。继承类对其父类的依赖性引入了额外的复杂性:对父类的更改会影响子类;当更改父类或子类中的任一方时,很难记住覆盖了哪些方法(或者没有覆盖哪些方法);而且是否应该调用相应的父类方法也不明朗。执行单一概念性任务的任何方法都应该是独立的,并应将其作为要重用的首选方法。要实现这一点,我们必须返回到过程式编程,将代码移出类实例方法并将其移入全局可见的过程中。为了提高这类过程的可重用性,您应该像编写静态实用方法那样编写这类方法:每个过程只使用其自身的输入参数和/或对其他全局可见过程的调用完成其工作,而且不应该使用任何非局部变量。这种外部依赖性的减弱降低了使用该过程的复杂性,从而可促进在别处对它的重用。当然,即便那些不计划重用的代码也会从这种结构中受益,因为它的结构总是相当清晰。在 Java 中,方法不能脱离类而存在。但是,您可以采取相关步骤,使方法成为单个类的、公共可见的静态方法。作为示例,您可以采用类似下面这样的一个类:class Polygon {..public int getPerimeter() {...}public boolean isConvex() {...}public boolean containsPoint(Point p) {...}..}并将其更改为类似以下的形式:class Polygon {..public int getPerimeter() {return pPolygon.computePerimeter(this);}public boolean isConvex() {return pPolygon.isConvex(this);}public boolean containsPoint(Point p) {return pPolygon.containsPoint(this, p);}..}其中,pPolygon 如下所示:class pPolygon {static public int computePerimeter(Polygon polygon) {...}static public boolean isConvex(Polygon polygon) {...}static public boolean containsPoint(Polygon polygon, Point p) {...}}类名 pPolygon 反映了该类所封装的过程主要与类型 Polygon 的对象有关。类名前的 p 表示该类的唯一用途就是将公共可见的静态过程组织起来。然而,在 Java 中类名以小写字母开头是不规范的,像 pPolygon 这样的类并不完成正常的类功能。这就是说,它不代表一类对象;它只是该语言所需的一个组织实体。在以上事例中所作更改的全部效果就是,客户端代码不再非要通过继承 Polygon 来重用其功能。现在这一功能在 pPolygon 类中是以过程为单位提供的。客户端代码仅使用它所需的功能,而不必关心它不需要的功能。这并不意味着类不会在新的过程式编程风格中发挥积极作用。恰恰相反,类要执行必要的分组任务,并封装它们所代表的对象的数据成员。此外,类通过实现多个接口而具备的多态性使其具备了卓越的可重用性,请参阅第二步中的说明。但是,您应该将通过类继承获得可重用性和多态性的方法归类到优先级较低的技术中,因为将功能包含在实例方法中并不是实现可重用性的最佳选择。四人合著的畅销书 Design Patterns 简要提及了一种与这一技术只有细微差别的技术。那本书中的 Strategy 模式提倡用一个共公接口将相关算法的每个系列成员都封装起来,以便客户端代码可互换这些算法。因为一种算法通常被编写为一个或几个独立的过程,因而这种封装强调重用执行单一任务(即一个算法)的过程,而不强调重用包含代码和数据、执行多项任务的对象。本步骤也体现了同样的基本思想。然而,用接口封装算法意味着将算法编写为实现该接口的一个对象。这意味着我们仍然被束缚在与数据耦合在一起的过程及其封装对象的其他方法上,因而使重用变得复杂。每次使用算法时必须实例化这些对象也是个问题,这将降低程序的性能。幸运的是, Design Patterns 提供的一种解决方案可解决这两个问题。在编写 Strategy 对象时您可使用 Flyweight 模式,以使每个对象仅有一个众所周知的共享实例(该实例处理执行问题),这样每个共享对象就不会在两次访问之间维护状态(因此该对象不包含任何成员变量,从而解决了许多耦合问题)。生成的 Flyweight-Strategy 模式将本步骤中封装功能的技术高度集成在全局可用的无状态过程中。第二步:将非基本数据类型的输入参数类型转换为接口类型通过接口参数类型而非通过类继承利用多态性,这是在面向对象编程方法中实现可重用性的真正基础,正如 Allen Holub 在 "Build User Interfaces for Object-Oriented Systems, Part 2" 中所讲的那样。“... 可重用性是通过编写接口,而不是通过编写类来实现的。如果一个方法的所有参数均为一些已知接口的引用,而这些接口又是由您从未听过的一些类实现的,那么该方法可对编写代码时还不存在的类的对象进行操作。从技术上讲,可重用的是方法,而不是传递给该方法的对象。”将 Holub 的论述应用到第一步的结果,一旦某个功能块可作为一个全局可见的独立过程,您就可以通过将它的每个类级输入参数类型转换为接口类型,从而进一步提高它的可重用性。这样,实现该接口类型的任何类的对象都符合该参数的要求,而不仅仅是符合原始类的要求。这样,该过程便潜在地可用于更多的对象类型。例如,假定您有一个全局可见的静态方法:static public boolean contains(Rectangle rect, int x, int y) {...}该方法旨在判断给定的矩形是否包含给定的位置。此处您应该将 rect 参数的类型从类类型 Rectangle 更改为接口类型,如下所示:static public boolean contains(Rectangular rect, int x, int y) {...}Rectangular could be the following interface:public interface Rectangular {Rectangle getBounds();}现在,可描述为 Rectangular 的类(即可实现 Rectangular 接口)的对象都可作为 rect 的参数传递给 pRectangular.contains()。我们通过放宽对可传递给方法的参数的约束来提高方法的可重用性。但是,就以上示例而言,当 Rectangle 接口的 getBounds 方法返回一个 Rectangle 时,您可能不知道使用 Rectangular 接口会有什么实际的好处;也就是说,如果我们知道我们要传入的对象在被请求时能返回 Rectangle;为什么不传入 Rectangle 类型而要传入接口类型呢?最重要的原因与集合有关。假定有这样一个方法:static public boolean areAnyOverlapping(Collection rects) {...}该方法旨在判断给定集合中的 rectangular 对象是否有重叠。接下来,在方法体中,当您依次处理集合中的每个对象时,如果无法将对象转换为诸如 Rectangular 这样的接口类型,如何才能访问那个对象的 rectangle 呢?唯一的选择是将对象转换为特定的类类型(我们已知该类中有一个方法能提供 rectangle),这意味着该方法必须事先知道它要对何种类类型进行操作,因此重用它时只能使用这些类型。这就是这一步首先要避免的问题!第三步:选择耦合性较小的输入参数接口类型在执行第二步时,应该选择何种接口类型来替代给定的类类型呢?答案是:能充分描述过程对参数的要求且累赘最少的任何接口。参数对象要实现的接口越小,任一特定类能实现该接口的机会就越大 -- 因而其对象可用作该参数的类的数量也就越多。很容易看出,如果您有如下这样一个方法:static public boolean areOverlapping(Window window1, Window window2) {...}该方法旨在判断两个(假定为 rectangular)窗口是否重叠,如果该方法仅要求它的两个参数提供它们各自的 rectangular 坐标,则最好简化这两个参数的类型以反映这一事实:static public boolean areOverlapping(Rectangular rect1, Rectangular rect2) {...}以上代码假定前面的 Window 类型对象也能实现 Rectangular。现在您就可以重用任何 rectangular 对象的第一个方法中所包含的功能。您可能有过多次这样的经历,即充分指定了参数要求的可用接口包含过多不必要的方法。碰到这种情况时,您就应在全局名称空间中定义一个新的公共接口,以便其他可能面临同样窘境的方法重用这个接口。您也可能有过多次这样的经历,即最好创建一个独特的接口来指定单个过程对一个参数的要求。您所创建的接口只会用于那个参数。当您希望将参数当作 C 中的函数指针处理时经常会出现这种情况,例如,假定有这样一个过程:static public void sort(List list, SortComparison comp) {...}该过程通过使用给定的比较对象 comp 对列表的所有对象进行比较,从而对给定的列表进行排序,sort 对 comp 的全部要求就是调用其单个方法执行比较。因此,SortComparison 应该是仅包含一个方法的接口:public interface SortComparison {boolean comesBefore(Object a, Object b);}该接口的唯一用途就是为 sort 提供一种访问完成其工作所需功能的方法,因此 SortComparison 不应在别处重用。小结以上三步旨在改进用更传统的面向对象方法编写的现有代码。将这三个步骤与面向对象编程结合使用即可构建一种新的方法,您可用这种新方法编写以后的代码,这样编写代码将提高方法的可重用性和内聚性,同时也会减少方法的相互耦合及复杂性。很明显,您不应该对本质上不适合重用的代码执行这些步骤。这种代码通常存在于程序的表示层。创建程序用户界面的代码及将输入事件绑定到完成实际操作的控制代码是不可重用的两个例子,因为它们的功能随程序的不同而相差甚远,根本无法实现可重用性。

时间: 2024-09-16 19:48:59

最大限制地提高代码的可重用性,克服传统面向对象编程方法在可重用性方面的不足的相关文章

最大限制地提高代码的可重用性

    重用是一种神话,这似乎正在日渐成为编程人员的一种共识.然而,重用可能难以实现,因为传统面向对象编程方法在可重用性方面存在一些不足.本技巧说明了组成支持重用的一种不同方法的三个步骤. 第一步:将功能移出类实例方法由于类继承机制缺乏精确性,因此对于代码重用来说它并不是一种最理想的机制.也就是说,如果您要重用某个类的单个方法,就必须继承该类的其他方法以及数据成员.这种累赘不必要地将要重用此方法的代码复杂化了.继承类对其父类的依赖性引入了额外的复杂性:对父类的更改会影响子类:当更改父类或子类中的

11步提高代码质量和整体工作效率

  这篇文章要介绍的,是我作为专业程序员这些年来学到的能真正提高我的代码质量和整体工作效率的11件事情. 1. 永远不要复制代码 不惜任何代价避免重复的代码.如果一个常用的代码片段出现在了程序中的几个不同地方,重构它,把它放到一个自己的函数里.重复的代码会导致你的同事 在读你的代码时产生困惑.而重复的代码如果在一个地方修改,在另外一个地方忘记修改,就会产生到处是bug,它还会使你的代码体积变得臃肿.现代的编程语 言提供了很好的方法来解决这些问题,例如,下面这个问题在以前很难解决,而如今使用lam

如何提高代码质量(管理篇):代码复查

也许你是一位项目经理,也许你是一位项目骨干成员,或者开发小组长.在我发表"如何提高代码质量"的这一系统文章后,有许多网友都向我抱怨,说他无法把握整个项目组成员的代码质量.我想,这也是所有项目组普遍存在的问题吧,它通常表现为以下几个问题: 软件项目普遍存在的问题 1)新手.任何项目组成员都不可避免地出现新手,他们往往是刚刚从大学毕业的学生.这些新手由于软件开发时间太短,往往技术不成熟,没有形成良好的开发习惯,所以编写代码质量较差,问题很多.他们常常成为项目组的"鸡肋"

嵌入式操作系统能提高代码可移植性吗?

问题描述 嵌入式操作系统能提高代码可移植性吗? 同一个平台,从高级芯片到低级芯片转移都这么麻烦,何况不同平台之间的转移.编写可移植的的代码是很重要的.嵌入式操作系统能够提高可移植性? 解决方案 不同平台,不同系统的嵌入代码,没什么可移植性. 当然,和写代码的人有关,像一些开源的项目,基本上不修改或很少的修改就可以跨平台的. 但这只是软件,而不是嵌入式系统.系统的代码,移植性不高! 解决方案二: 可移植性有几个层次: 源代码的可移植性(不修改源代码,或者只修改很少的代码就可以移植到另一个平台) A

利用Java反射(Reflection) 机制提高代码的行覆盖率

在本文中,您将看到如何通过使用反射机制,在外部直接对目标类中的不可访问成员进行测试,以提高被测代码数量:以及通过修改 Cobertura 源码,使其支持通过正则表达式来过滤不需要进行单元测试的代码,以降低代码总数.代码覆盖率的提高,减少了单元测试过程中未被覆盖的代码数量,降低了http://www.aliyun.com/zixun/aggregation/7155.html">开发人员编写或修改单元测试用例的时间成本,从而提高了整个单元测试的效率. 单元测试是软件开发过程中重要的质量保证环

关于如何提高代码可测试性的一些看法

在这些条件下,伪造一个RGHConnection对象最好的方法是对RGHConnection类应用接口提取.如果你手头一个支持重构的工具,那么它很可能也会支持接口提取方法.我们来看一下接口提取后的情况: <interface>  IRGHConnection  +connect()  + disconnect()  +RFDIReportFor(id:int):RFDIReport  +ACTIOReportFor(customerID:int) ACTIOReport 由于retry()和f

提高代码质量:如何编写函数

提高代码质量:如何编写函数 函数是实现程序功能的最基本单位,每一个程序都是由一个个最基本的函数构成的.写好一个函数是提高程序代码质量最关键的一步.本文就函数的编写,从函数命名,代码分布,技巧等方面入手,谈谈如何写好一个可读性高.易维护,易测试的函数. 命名 首先从命名说起,命名是提高可读性的第一步.如何为变量和函数命名一直是开发者心中的痛点之一,对于母语非英语的我们来说,更是难上加难.下面我来说说如何为函数命名的一些想法和感受: 采用统一的命名规则 在谈及如何为函数取一个准确而优雅的名字之前,首

如何提高代码性能?

问题描述 做开发2年多了,实现功能已不是问题,接下来的目标就是用最短的代码,写出优质.性能高的代码,关于如何提高自己代码质量,大家都有哪些经验呢? 解决方案 解决方案二:引用楼主zhenzhenzhao12的回复: 接下来的目标就是用最短的代码,写出优质.性能高的代码 我不会定这种目标.这没有意义.要替高性能,必需先要找到用户体验为根据,搞明白"到底一个操作需要多少毫秒"为指标.要拿出"具体数据来"才行动,不要追求什么"越快越好"这种空洞的.不具

提高代码可读性的十大注释技巧分享_相关技巧

本文讲述了提高代码可读性的十大注释技巧.分享给大家供大家参考,具体如下: 很多程序员在写代码的时候往往都不注意代码的可读性,让别人在阅读代码时花费更多的时间.其实,只要程序员在写代码的时候,注意为代码加注释,并以合理的格式为代码加注释,这样就方便别人查看代码,也方便自己以后查看了.下面分享十个加注释的技巧: 1. 逐层注释 为每个代码块添加注释,并在每一层使用统一的注释方法和风格.例如: 针对每个类:包括摘要信息.作者信息.以及最近修改日期等: 针对每个方法:包括用途.功能.参数和返回值等. 在