内容简介:不完美的库类(Incomplete Library Class)当一个类库已经不能满足实际需要时,你就不得不改变这个库(如果这个库是只读的,那就没辙了)。许多编程技术都建立在库类的基础上。库类的作者没用未卜先知的能力,不能因此责怪他们。麻烦的是库往往构造的不够好,而且往往不可能让我们修改其中的类以满足我们的需要。
不完美的库类
不完美的库类(Incomplete Library Class)
当一个类库已经不能满足实际需要时,你就不得不改变这个库(如果这个库是只读的,那就没辙了)。
问题原因
许多编程技术都建立在库类的基础上。库类的作者没用未卜先知的能力,不能因此责怪他们。麻烦的是库往往构造的不够好,而且往往不可能让我们修改其中的类以满足我们的需要。
解决方法
引入外加函数(Introduce Foreign Method) 引入本地扩展(Introduce Local Extension)
收益
- 减少代码重复(你不用一言不合就自己动手实现一个库的全部功能,代价太高)
何时忽略
- 如果扩展库会带来额外的工作量。
重构方法说明
引入外加函数(Introduce Foreign Method)
问题
你需要为提供服务的类增加一个函数,但你无法修改这个类。
class Report { //... void sendReport() { Date nextDay = new Date(previousEnd.getYear(), previousEnd.getMonth(), previousEnd.getDate() + 1); //... } }复制代码
解决
在客户类中建立一个函数,并一个第一个参数形式传入一个服务类实例。
class Report { //... void sendReport() { Date newStart = nextDay(previousEnd); //... } private static Date nextDay(Date arg) { return new Date(arg.getYear(), arg.getMonth(), arg.getDate() + 1); } }复制代码
引入本地扩展(Introduce Local Extension)
问题
你需要为服务类提供一些额外函数,但你无法修改这个类。
解决
建立一个新类,使它包含这些额外函数,让这个扩展品成为源类的子类或包装类。
中间人
中间人(Middle Man)
如果一个类的作用仅仅是指向另一个类的委托,为什么要存在呢?
问题原因
对象的基本特征之一就是封装:对外部世界隐藏其内部细节。封装往往伴随委托。但是人们可能过度运用委托。比如,你也许会看到一个类的大部分有用工作都委托给了其他类,类本身成了一个空壳,除了委托之外不做任何事情。
解决方法
应该运用 移除中间人(Remove Middle Man)
,直接和真正负责的对象打交道。
收益
- 减少笨重的代码。
何时忽略
如果是以下情况,不要删除已创建的中间人:
- 添加中间人是为了避免类之间依赖关系。
- 一些 设计模式 有目的地创建中间人(例如代理模式和装饰器模式)。
重构方法说明
移除中间人(Remove Middle Man)
问题
某个类做了过多的简单委托动作。
解决
让客户直接调用委托类。
依恋情结
依恋情结(Feature Envy)
一个函数访问其它对象的数据比访问自己的数据更多。
问题原因
这种气味可能发生在字段移动到数据类之后。如果是这种情况,你可能想将数据类的操作移动到这个类中。
解决方法
As a basic rule, if things change at the same time, you should keep them in the same place. Usually data and functions that use this data are changed together (although exceptions are possible).
有一个基本原则:同时会发生改变的事情应该被放在同一个地方。通常,数据和使用这些数据的函数是一起改变的。
搬移函数(Move Method) 提炼函数(Extract Method) 提炼函数(Extract Method)
收益
- 减少重复代码(如果数据处理的代码放在中心位置)。
- 更好的代码组织性(处理数据的函数靠近实际数据)。
何时忽略
- 有时,行为被有意地与保存数据的类分开。这通常的优点是能够动态地改变行为(见策略设计模式,访问者设计模式和其他模式)。
重构方法说明
搬移函数(Move Method)
问题
你的程序中,有个函数与其所驻类之外的另一个类进行更多交流:调用后者,或被后者调用。
解决
在该函数最常引用的类中建立一个有着类似行为的新函数。将旧函数变成一个单纯的委托函数,或是旧函数完全移除。
提炼函数(Extract Method)
问题
你有一段代码可以组织在一起。
void printOwing() { printBanner(); //print details System.out.println("name: " + name); System.out.println("amount: " + getOutstanding()); }复制代码
解决
移动这段代码到一个新的函数中,使用函数的调用来替代老代码。
void printOwing() { printBanner(); printDetails(getOutstanding()); } void printDetails(double outstanding) { System.out.println("name: " + name); System.out.println("amount: " + outstanding); }复制代码
狎昵关系
狎昵关系(Inappropriate Intimacy)
一个类大量使用另一个类的内部字段和方法。
问题原因
类和类之间应该尽量少的感知彼此(减少耦合)。这样的类更容易维护和复用。
解决方法
- 最简单的解决方法是运用
搬移函数(Move Method)
和搬移字段(Move Field)
来让类之间斩断羁绊。
-
你也可以看看是否能运用
将双向关联改为单向关联(Change Bidirectional Association to Unidirectional)
让其中一个类对另一个说分手。 -
如果这两个类实在是情比金坚,难分难舍,可以运用
提炼类(Extract Class)
把二者共同点提炼到一个新类中,让它们产生爱的结晶。或者,可以尝试运用隐藏委托关系(Hide Delegate)
让另一个类来为它们牵线搭桥。 -
继承往往造成类之间过分紧密,因为子类对超类的了解总是超过后者的主观愿望,如果你觉得该让这个子类自己闯荡,请运用
以委托取代继承(Replace Inheritance with Delegation)
来让超类和子类分家。
收益
- 提高代码组织性。
- 提高代码复用性。
重构方法说明
搬移函数(Move Method)
问题
你的程序中,有个函数与其所驻类之外的另一个类进行更多交流:调用后者,或被后者调用。
解决
在该函数最常引用的类中建立一个有着类似行为的新函数。将旧函数变成一个单纯的委托函数,或是旧函数完全移除。
搬移字段(Move Field)
问题
在你的程序中,某个字段被其所驻类之外的另一个类更多地用到。
解决
在目标类新建一个字段,修改源字段的所有用户,令他们改用新字段。
将双向关联改为单向关联(Change Bidirectional Association to Unidirectional)
问题
两个类之间有双向关联,但其中一个类如今不再需要另一个类的特性。
解决
去除不必要的关联。
提炼类(Extract Class)
问题
某个类做了不止一件事。
解决
建立一个新类,将相关的字段和函数从旧类搬移到新类。
隐藏委托关系(Hide Delegate)
问题
客户通过一个委托类来调用另一个对象。
解决
在服务类上建立客户所需的所有函数,用以隐藏委托关系。
以委托取代继承(Replace Inheritance with Delegation)
问题
某个子类只使用超类接口中的一部分,或是根本不需要继承而来的数据。
解决
在子类中新建一个字段用以保存超类;调整子类函数,令它改而委托超类;然后去掉两者之间的继承关系。
过度耦合的消息链
过度耦合的消息链(Message Chains)
消息链的形式类似于: obj.getA().getB().getC()
。
问题原因
如果你看到用户向一个对象请求另一个对象,然后再向后者请求另一个对象,然后再请求另一个对象……这就是消息链。实际代码中你看到的可能是一长串 getThis()或一长串临时变量。采取这种方式,意味客户代码将与查找过程中的导航紧密耦合。一旦对象间关系发生任何变化,客户端就不得不做出相应的修改。
解决方法
- 可以运用
隐藏委托关系(Hide Delegate)
删除一个消息链。
- 有时更好的选择是:先观察消息链最终得到的对象是用来干什么的。看看能否以
提炼函数(Extract Method)
把使用该对象的代码提炼到一个独立函数中,再运用搬移函数(Move Method)
把这个函数推入消息链。
收益
- 能减少链中类之间的依赖。
- 能减少代码量。
何时忽略
- 过于侵略性的委托可能会使 程序员 难以理解功能是如何触发的。
重构方法说明
隐藏委托关系(Hide Delegate)
问题
客户通过一个委托类来调用另一个对象。
解决
在服务类上建立客户所需的所有函数,用以隐藏委托关系。
提炼函数(Extract Method)
问题
你有一段代码可以组织在一起。
void printOwing() { printBanner(); //print details System.out.println("name: " + name); System.out.println("amount: " + getOutstanding()); }复制代码
解决
移动这段代码到一个新的函数中,使用函数的调用来替代老代码。
void printOwing() { printBanner(); printDetails(getOutstanding()); } void printDetails(double outstanding) { System.out.println("name: " + name); System.out.println("amount: " + outstanding); }复制代码
搬移函数(Move Method)
问题
你的程序中,有个函数与其所驻类之外的另一个类进行更多交流:调用后者,或被后者调用。
解决
在该函数最常引用的类中建立一个有着类似行为的新函数。将旧函数变成一个单纯的委托函数,或是旧函数完全移除。
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网
猜你喜欢:- 如何提高golang的可读性
- 可读性代码:为什么、怎样以及什么时候
- 代码注释的艺术,再也不怕被说代码可读性差啦!
- [译] 作为一个 Python 爱好者,如何写出高可读性的代码?
- 内聚代码提高逻辑可读性,用MCVP接续你的大逻辑
- 跟我学Spring之自定义bean容器提升代码可读性
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
Web标准和SEO应用实践
Aarron Walter / 李清 / 机械工业出版社 / 2008 / 36.00元
本书是关于搜索引擎优化和易发现性的技术指南。. 本书介绍Web标准、可访问性以及Ajax、API、Flash和微格式等内容,包括标记策略、服务器端策略、内容策略、建构易发现的博客、在网站内添加搜索、防止易发现性障碍、用邮件列表挽回流量、将易发现性付诸实践。 本书适合网站开发者与SEO技术业余爱好者等参考。 这不是为营销专家写的一本SEO的书。 针对那些想要找到网站的目标用户......一起来看看 《Web标准和SEO应用实践》 这本书的介绍吧!