重构-改善既有代码的设计之1

来源:互联网 发布:ubuntu 开启端口 编辑:程序博客网 时间:2024/06/10 15:35

6、重新组织函数

函数层面

Extract Method(提炼函数)
——单一原则
Inline Method(内联函数)
——在函数调用点插入函数本体,然后移除该函数

Tips:间接层的使用需要合理

临时变量

Replace Temp With Query(以查询代替临时变量)
——临时变量只使用一次,或者赋值给临时变量的表达式不受其他条件的影响
Introduce Explaining Variable(引入解释性变量)
——将复杂表达式的结果放进一个临时变量中,以此变量名来解释表达式的用途(特别在条件语句中)
Split Temporary Variable(分解临时变量)
——针对每次赋值,创造出一个独立、对应的临时变量

参数问题

Remove Assignments to Parameters(移除对参数的赋值)
——不对入参进行赋值

其他

Replace Method With Method Object(以函数对象取代函数)
将这个函数放进一个对象中,这样,该函数内的局部变量就可以变为对象的公有变量,可以在该对象中将函数进行拆分。

7、在对象中搬移特性

函数层面

Move Method(搬移函数)
——某类中的函数更多的被其他类调用和被调用
——在其他类中建立一个有类似行为的新函数,将旧函数写成一个单纯的委托函数,或者直接删除
Move Field(搬移字段)
——在目标类新建一个字段,修改源字段的所有用户,令他们改成新字段
Extract Class(提炼类)
——某个类做了应该由两个类做的事
——建立一个新类,将相关字段和函数从旧类迁移到新类中
Inline Class(将类内联化)
——某个类没有做太多的事情
——将这个类的所有特性搬移到另一个类中,然后移除原类
Hide Delegate(隐藏委托关系)
——客户通过一个委托类来调用另一个对象
——在服务类上建立一个客户所需的函数,然后该函数调用另外一个对象,即客户只调用服务类
Remove Middle Man(移除中间人)
——客户直接调用受托类,取消中间的委托类
Introduce Local Extension(引入本地扩展)
——需要为服务类提供一些额外的函数,但无法修改这个类
——建立一个新类,使他包含这些额外的函数,让这个扩展类成为源类的子类或者包装类

8、重新组织数据

Self Encapsulate Field(自封装字段)
——某个字段和字段之间的耦合关系比较复杂
——为这个字段建立set和get函数,并且只以这些函数来访问字段
Replace Data Value With Object(以对象取代数据值)
——你有一个数据项,需要和其他数据和行为一起使用才有意义
——将数据项变为对象
Change Value to Reference(将值对象改为引用对象)
——从一个类衍生出许多彼此相同的实例,将他们替换为同一个对象
——将这个值对象变成引用对象
Change Reference to Value(将引用对象改为值对象)
——一个引用对象,很小且不可变
——将它变成一个值对象
Replace Array With Object(以对象取代数组)
——数组中的各元素各自代表不同的东西
——以对象替代数组,对数组中的每个元素,以一个字段来表示
Duplicate Observed Data(复制被监视数据)
——将数据复制到一个领域对象中。建立一个观察者模式,用于同步这些数据。
Change Undirectional Association to Bidirectional(将单向关联改为双向关联)
——两个类都需要使用对方特性,但其间只有一条单向连接
——增加一个反向指针,并使修改函数能够同时更新两条连接
Change Bidirectional Association to Undirectional(将双向关联改为单向关联)
——两个类之间有双向关联,但其中一个类如今不再需要另一个类的特性
——去除不需要的关联
Replace magic Number With Symbolic Constant(以字面常量取代魔法数)
——创造一个常量,将字面数值替换为这个常量
Encapsulate Field(封装字段)
——将类中的public字段声明为private,并提供相应的访问函数
Encapsulate Collection(封装集合)
——有一个函数返回一个集合
——让这个函数返回该集合的一个只读副本,并在这个类中提供添加和移除的函数
Replace Record With Data Class(以数据类取代记录)
——为该记录创建一个数据对象
Replace Type Code With Code(以类取代类型码)
——类之中有一个数值类型码,但他不影响类的行为
——以一个新的类替换该数值类型码
Replace Type Code With Subclasses(以子类取代类型码)
——类之中有一个数值类型码,但他影响类的行为
——以子类取代这个类型码
Replace Type Code With State/Strategy(以状态取代类型码)
——类之中有一个数值类型码,但他影响类的行为,但无法通过继承的方法消除它
——以状态对象取代类型码
Replace Subclass With Fields(以字段取代子类)
——子类的唯一差别就是在返回常量数据的函数上
——修改这些函数,使他们返回超类中的某个字段,销毁子类

9、简化条件表达式

Decompose Condition(分解条件表达式)
——在if、else段落中分别提炼出独立的函数
Consolidate Conditional Expression(合并条件表达式)
——将这些表达式合并为一个条件表达式,并将这个表达式提炼为一个独立的函数
Consolidate Duplicate Conditional Fragments(合并重复的条件判断)
——在条件表达式的每个分支上都有相同的代码
——将这个重复代码搬移到条件表达式之外
Remove Control Flag(移除控制标记)
——以break和return作为控制标记
Replace Conditional With Polymorphism(以多态取代条件表达式)
——将表达式的每个分支放进一个子类内的复写函数中,然后将原始函数声明为抽象函数
Introduce Null Object(引入Null对象)
——需要检查某个对象是否为null
——将null值替换为null对象
Introduce Asseration(引入断言)
——某一段代码需要对程序的状态做出某种假设
——以断言明确表现这种假设

10、简化函数调用

Rename Method(函数改名)
——函数名称未能解释函数
——修改对应的函数名称
Add Parameter(添加参数)
——某个函数需要从调用端得到更多的信息
——为函数添加一个对象/数组参数,让该参数带入函数中,方便扩展
Remove Parameter(移除参数)
——函数本体不再需要某个参数
——将该参数去除
Separate Query from Modifier(将查询函数和修改函数分离)
——某个函数即返回对象的状态值,有修改对象状态
——单一原则,差分成两个不同的函数
Parameterize Method(令函数携带参数)
——多个函数实现类似的功能
——将该函数增加参数,然后通过参数来处理变化情况
Replace Parameter With Exolicit Method(以明确函数取代参数)
——不同的参数值采用不同的行为
——针对该参数的每一个值,建立一个独立的函数
Preserve Whole Object(保持对象完整)
——从对象/数组中去除若干值,作为入参
——直接传递整个对象/数组(减少了入参项,参数更稳固)
Replace Parameter With Method(以函数取代参数)
——某个入参可以通过其他途径获取,即在函数内或函数外取值都是一样的情况
——去除这个入参,在函数内用方法获取这个入参
——只有在必要关头才添加参数,预先添加惨回溯很可能不是你所需要的
Introduce Parameter Object(引入参数对象)
——特定的一组参数总是一起被传递(数据泥团Data Clumps)
——运用一个对象包装这些数据,然后用该对象/数组取代他们
Remove Setting Method(移除设置函数)
——某个字段应该再对象创建时被设值,然后不再改变
——去掉该字段的所有设值函数
Hide Method(隐藏函数)
——有一个函数,从来没有被其他任何类用到
——将这个函数修改为private(修改函数的可见度)
Replace Constructoe With Factory Method(以工厂函数取代构造函数)
——在创建对象时不仅仅是做简单的建构行动
——工厂函数根据参数的个数和类型,选择不同的创建行为
Encapsulate Downcast(封装向下转型)
——某函数返回的对象,需要由调用者执行向下转型
——将向下转型动作移到函数中
Replace Errcode With Exception(以异常取代错误码)
——某个函数返回一个特定的代码,表示某种错误情况
——改用异常
Replace Exception With Test(以测试取代异常)
——面对一个调用者可以预先检查的条件,你抛出了一个异常
——修改调用者,使他在调用函数之前先做一个检查

11、处理概括关系

Pull Up Field(字段上移)
——两个子类拥有相同的字段
——将该字段移至超类
Pull Up Method(函数上移)
——两个子类拥有相同的函数
——将该函数移至超类
Pull Up Constructor Body(构造函数上移)
——各个子类中拥有一些构造函数,他们的本体几乎一致
——在超类中新建一个构造函数,并在子类构造函数中调用它
Push Down Method(函数下移)
——超类中的某个函数只与部分子类有关
——将这个函数移到相关的子类中去
Push Down Field(字段下移)
——超类中的某个字段只与部分子类有关
——将这个字段移到相关的子类中去
Extract Subclass(提炼子类)
——若干客户使用类接口中的统一子集,或者两个类中的接口有相同的部分
——将相同的子集提炼到一个独立的接口中
Extract Interface(提炼接口)
——两个类有相似特性
——为两个类建立一个超类,将相同的特性移至超类
Extract Superclass(提炼超类)
——两个类有相似特性
——为两个类建立一个超类,将相同的特性移至超类
Collapse Hierarchy(折叠继承体系)
——超类和子类之间无太大区别
——将他们合为一体
Form Template Method(塑造模板函数)
——你有一些子类,其中相应的函数以相同的顺序执行类似的操作,但各个操作的细节有所不同
——将这些操作分别放在独立的函数中,并保持他们都有相同的签名,于是原函数变得相同了。然后将原函数上移至超类。
Replace Inheritance With Delegation(以委托取代继承)
——某个子类只使用超类接口中的一部分
——将子类函数改为委托超类,去掉两者之间的继承关系
Replace Delegation With Inheritance(以继承取代委托)
——你在两个类之间使用委托,并经常为整个接口编写很多简单的委托函数
——让委托类继承受托类

12、大型重构

Tease Apart Inheritance(梳理并分解继承体系)
——某个继承体系同时承担两项责任
——建立两个继承体系,并通过委托关系让其中一个可以调用另一个
Convert Procedural Design to Objects(以过程化设计转化为对象设计)
——将数据记录变成对象,将大块的行为分为小块,并将行为移入相关对象中
Extract Hierarchy(提炼继承体系)
——某个类做了太多的工作,其中一部分是以大量的条件表达式完成的
——建立继承体系,以一个子类表示一种特殊的情况

0 0
原创粉丝点击