重构列表

[1] 重新组织你的函数

1.1 Extract Method(提炼函数

有一段代码可以被组织在一起并独立出来,将这段代码放进一个独立函数中,并让函数名称解释该函数的用途。

1.2 Inline Method(将函数内联化

一个函数的本体与名称同样清楚易懂。在函数的调用点插入函数本体,然后移除该函数。

1.3 Inline Temp(将临时变量内联化

有一个临时变量,只被一个简单表达式赋值一次,而它妨碍了其他重构手法。将所有对该变量的引用动作,替换为对它赋值的那个表达式自身。

1.4 Replace Temp With Query(以查询取代临时变量

以一个临时变量保存某一表达式的运算结果。将这个表达式提炼到一个独立函数中。将这个临时变量的所有引用点替换为对新函数的调用。此后,新函数就可以被其他函数使用。

1.5 Introduce Explaining Variable(引入解释性变量

程序中有一个复杂表达式。将该复杂表达式(或其中一部分)的结果放进一个临时变量,以此变量名称来解释表达式用途。

1.6 Split Temporary Variable(分解临时变量

有某个临时变量被赋值超过一次,它既不是循环变量,也不被用于收集计算结果。针对每次赋值,创造一个独立、对应的临时变量。

1.7 Remove Assignments to Paramete (移除对参数的赋值动作

代码对一个参数进行赋值。以一个临时变量取代该参数的位置。

1.8 Replace Method with Method Object(以函数对象取代函数

有一个大型函数,其中对局部变量的使用使你无法采用[1.1提炼函数]。

将这个函数放进一个单独对象中,如此一来局部变量就成了对象内的字段。

然后,你可以在同一个对象中将这个大型函数分解为多个小型函数。

1.9 Substitute Algorithm(替换你的算法

想要把某个算法替换为另一个更清晰的算法。将函数本体替换为另一个算法。


[2] 在对象之间移动特性

2.1 Move Method(搬移函数

在程序中,如果有个函数与其所驻之类的另外一个类进行更多交流,调用后者或被后者调用。

在该函数最常引用的类中建立一个有着类似行为的新函数。

将旧函数变成一个单纯的委托函数,或将旧函数完全移除。

2.2 Move Field(搬移字段

程序中,某个字段被其所驻类之外的另一个类更多的用到。在目标类新建一个字段,修改源字段的所有用户,令他们改用新字段。

如果发现,对于一个字段,在其所驻类之外的另一个类中有更多函数使用了它,我就会考虑搬移这个字段。

2.3 Extract Class(提炼类

某个类做了应该由两个类做的事。建立一个新类,将相关的字段和函数从旧类搬移到新类。

2.4 Inline Class(将类内联化

如果某个类没有做太多事情,将这个类中的所有特性移到另外一个类中,然后删除原类。

2.5 Hide Delegate(隐藏「委托关系」)

客户通过一个委托类来调用另一个对象。

在服务类上建立客户所需的所有函数,用以隐藏委托关系。

2.6 Remove Middle Man(移除中间人

某个类做了过多简单委托工作;让客户直接调用委托类。

2.7 Introduce Foreign Method(引入外加函数

你需要为提供服务的类增加一个函数,但你无法修改这个类。在客户类中建立一个函数,并以第一参数形式传入一个服务类实例。

2.8 Introduce Local Extension(引入本地扩展(子类或包装类))

你需要为服务类提供一些额外函数,但你无法修改这个类。建立一个新类,使它包含这些额外函数。让这个扩展品成为源类的子类或包装类。

 

[3] 重新组织你的数据

3.1 Self Encapsulate Field(自封装字段

直接访问一个字段,但与字段之间的耦合关系逐渐变得笨拙。为这个字段建立取值/设值函数,并且只以这些函数来访问字段。

3.2 Replace Data Value with Object(以对象取代数据值

有一个数据项,需要与其他数据和行为一起使用才有意义。将数据项变为对象。

3.3 Change Value to Reference(将实值对象改为引用对象

从一个类衍生出许多彼此相等的实例,希望将它们替换为同一个对象。将这个值对象变为引用对象。

3.4 Change Reference to Value(将引用对象改为实值对象

有一个引用对象,很小且不可变,而且不易管理、将它变成一个值对象。

3.5 Replace Array with Object(以对象取代数组

有一个数组 ,其中的元素各自代表不同的东西。以对象替换数组。对于数组中的每个元素,以一个字段来表示。

3.6 Duplicate Observed Data(复制「被监视数据」)

有一些领域数据置身于GUI控件中,而领域函数需要访问这些数据。将该数据复制到一个领域对象中。

建立一个Observer模式,用以同步领域对象和GUI对象内的重复数据。

3.7 Change Unidirectional Association to Bidirectional(将单向关联改为双向

两个类都需要使用对方特性,但其间只有一条单向连接。添加一个反向指针,并使修改函数能够同时更新两条连接。

3.8 Change Bidirectional Association to Unidirectional(将双向关联改为单向

两个类之间有双向关联,但其中一个类如今不再需要另一个类的特性。去除不必要的关联。

3.9 Replace Magic Number with Symbolic Co tant (以符号常量/字面常量取代魔法数

有一个字面数值,带有特别含义。创造一个常量,根据其意义为它命名,并将上述的字面数值替换为这个常量。

3.10 Encapsulate Field(封装字段

类中存在一个public字段,将它声明为private,并提供相应的访问函数。

3.11 Encapsulate Collection(封装集合

有个函数返回一个集合。让这个函数返回该集合的一个只读副本,并在这个类中提供添加/移除集合元素的函数。

3.12 Replace Record with Data Class(以数据类取代记录

你需要面对传统编程环境中的记录结构。为该记录创建一个“哑”数据对象。

3.13 Replace Type Code with Class(以类取代型别码

类中有一个数值类型码,但它并不影响类的行为。以一个新的类替换该数值类型码。

3.14 Replace Type Code with Subclasses (以子类取代类型码

有一个不可变的类型码,它会影响类的行为。以子类取代这个类型码。

3.15 Replace Type Code with State/Strategy (以State/Strategy状态对象取代类型码

有一个类型码,它会影响类的行为。但你无法通过继承手法消除它。以状态对象取代类型码。

3.16 Replace Subclass with Fields(以字段取代子类

各个子类的唯一差别只在”返回常量数据“的函数身上。修改这些函数,使它们返回超类中的某个(新增)字段,然后销毁子类。

 

[4] 简化条件表达式

4.1 Decompose Conditional(分解条件式

有一个复杂的条件(if-then-else)语句。从if、then、else三个段落中分别提炼出独立函数。

4.2 Co olidate Conditional Expression(合并条件式

有一系列条件测试,都得到相同的结果。将这些测试合并为一个条件表达式,并将这个条件表达式提炼成为一个独立函数。

4.3 Co olidate Duplicate Conditional Fragments (合并重复的条件执行片段

在条件表达式的每个分支上有着相同的一段代码。将这段重复代码搬移到条件表达式之外。

4.4 Remove Control Flag(移除控制标记

在一系列布尔表达式中,某个变量带有“控制标记”(control flag)的作用。以break语句或return语句取代控制标记。

4.5 Replace Nested Conditional with Guard Clauses (以卫语句取代嵌套条件式

函数中的条件逻辑使人难以看清正常的分支执行路径。使用卫语句表现所有特殊情况。所

谓卫语句,如果某个条件极其罕见,就应该单独检查该条件,并在该条件为真时立刻从函数中返回。这样的单独检查常常被称为“卫语句”。

4.6 Replace Conditional with Polymorphism(以多态取代条件式

有个条件表达式,它根据对象类型的不同而选择不同的行为。

将这个条件表达式的每个分支放进一个子类内的覆写函数中,然后将原始函数声明为抽象函数。

4.7 Introduce Null Object(引入NULL对象

你需要再三检查某个对象是否为NULL。将NULL值替换为NULL对象。

4.8 Introduce Assertion(引入断言)

某一段代码需要对程序状态做出某种假设。以断言明确表现这种假设。

 

[5] 简化函数调用

5.1 Rename Method(重新命名函数)

函数名称未能揭示函数的用途。修改函数名称。

5.2 Add Parameter(添加参数)

某个函数需要从调用端得到更多的信息。为此函数添加一个对象参数,让该对象带进函数所需的信息。

5.3 Remove Parameter(移除参数)

函数本体不再需要某个参数。将该参数去除。

5.4 Separate Query from Modifier(将查询函数和修改函数分离)

某个函数既返回对象状态值,又修改对象状态。

建立两个不同的函数,其中一个负责查询,另一个负责修改。

5.5 Parameterize Method(令函数携带参数)

若干函数做了类似的工作,但在函数本体中却包含了不同的值。

建立单一函数,以参数表达那些不同的值。

5.6 Replace Parameter with Explicit Methods(以明确函数取代参数)

有一个函数,其中实现完全取决于参数值而采取不同行为。

针对该参数的每一个可能值,建立一个独立函数。

5.7 Preserve Whole Object(保持对象完整)

从某个对象中取出若干值,将它们作为某一次函数调用时的参数。改为传递整个对象。

5.8 Replace Parameter with Method(以成员函数取代参数)

一个对象调用某个函数,并将所得结果作为参数,传递给另一个函数,而接受该参数的函数本身也能够调用前一个函数。

让参数接受者去除该项参数,并直接调用前一个函数。

5.9 Introduce Parameter Object(引入参数对象)

某些参数总是很自然地同时出现。以一个对象取代这些参数。

5.10 Remove Setting Method(移除设值函数)

类中的某个字段只在对象创建时被设值,然后就不再改变。去掉该字段的所有设值函数。

5.11 Hide Method(隐藏你的函数)

类中有一个函数,从来没有被其他任何类用到。将这个函数修改为private私有访问权限。

5.12 Replace Co tructor with Factory Method(以工厂方法取代构造函数)

你希望在创建对象时不仅仅是做简单的构建动作。将构造函数替换为工厂函数。

5.13 Encapsulate Downcast(封装「向下转型」动作)

某个函数返回的对象,需要由函数调用者执行向下转型。

那么,将向下转型动作移到函数中(即封装)。这样函数调用者不用再考虑转换对象类型。

5.14 Replace Error Code with Exception(以异常取代错误码)

某个函数返回一个特定的代码,用以表示某种错误情况。改用异常。

5.15 Replace Exception with Test(以测试取代异常

面对一个调用者可以预先检查的条件,你抛出了一个异常。修改调用者,使它在调用函数之前先做检查。

 

[6] 处理概括关系

6.1 Pull Up Field(字段上移)

即成员变量上移。两个子类拥有相同的字段,将该字段移至基类。

6.2 Pull Up Method(函数上移)

即成员函数上移。两个子类拥有相同的函数(产生完全相同的结果),将该函数移至基类。

6.3 Pull Up Co tructor Body(构造函数本体上移)

在各个子类中拥有一些构造函数,它们的本体几乎完全一致。在基类中新建一个构造函数,并在子类构造函数中调用它。

6.4 Push Down Method(函数下移)

即成员函数下移。基类中某个函数只与部分(而非全部)子类有关,将这个函数移到相关的那些子类中去。

6.5 Push Down Field(字段下移)

即成员变量。基类中某个字段只与部分(而非全部)子类有关,将这个字段移到相关的那些子类中去。

6.6 Extract Subclass(提炼子类)

类中的某些特性只被某些(而非全部)实例用到。新建一个子类,将上面所说的那一部分特性移到子类中。

6.7 Extract Superclass(提炼超类)

两个类有相似特性。为这两个类建立一个基类,将相同特性移至基类。

6.8 Extract Interface(提炼接口)

若干客户使用类接口中的同一子集,或者两个类的接口有部分相同。将相同的子集提炼到一个独立接口中。

6.9 Collapse Hierarchy(折叠继承体系)

基类和子类无太大区别,将它们合为一体。

6.10 Form Template Method(塑造模板函数)

有一些子类,其中相应的某些函数以相同顺序执行类似的操作,但各个操作的细节上稍有不同。

将这些细节稍不同的操作分别放进独立的函数中,并保持它们都有相同的函数名,于是原函数也就变得相同了。

然后,将原函数上移至基类中。

6.11 Replace Inheritance with Delegation(以委托取代继承

某个子类只使用基类接口中一部分,或者根本不需要继承而来的数据。在子类中新建一个字段用以保存基类:调整子类函数,令它改而委托基类。然后去掉两者之间的继承关系。

6.12 Replace Delegation with Inheritance(以继承取代委托

在两个类之间使用委托关系,并经常为整个接口编写许多极其简单的委托函数。让委托类继承受托类。


[7] 大型重构

7.1 Tease Apart Inheritance(疏理并分解继承体系)

7.2 Convert Procedural Design to Objects(将过程化设计转化为对象设计)

7.3 Separate Domain from Presentation(将领域和表述/显示分离)

7.4 Extract Hierarchy(提炼继承体系)

 

Good Good Study, Day Day Up.

顺序 选择 循环 总结

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值