第七章 在对象之间搬移特性

本文探讨了面向对象设计中的几种重构技术,包括Move Method(搬移函数)、Move Field(搬移字段)、Extract Class(提炼类)、Inline Class(将类内联化)、Hide Delegate(隐藏委托关系)、Remove Middle Man(移除中间人)、Introduce Foreign Method(引入外加函数)和Introduce Local Extension(引入本地扩展)。这些技术旨在改善代码结构,减少耦合,提高类的内聚性和职责明确性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


在对象的设计过程中,“决定把责任放在哪儿”即使不是最重要的事,也是最重要的事之一。类往往会因为承担过多责任而变得臃肿不堪。

Move Method(搬移函数)

  • 现象: 程序中,有个函数与其所在类之外的另一个类进行更多交流:调用后者,或被后者调用
  • 做法:在该函数最常引用的类中建立一个有着类似行为的新函数,将旧函数变成一个单纯的委托函数,或是将旧函数完全移除。
  • 一个类有太多行为,或如果一个类与另一个类有太多合作而形成高度耦合,就要搬移函数,从而使系统中的类更简单,更干净利落地实现系统交付的任务
  • 当目标类需要使用源类的特性时,有4种选择:
    • 将这个特性也移到目标类;
    • 建立或使用一个从目标类到源类的引用关系;
    • 将源对象当做参数传给目标函数;
    • 如果所需特性是个变量,将它作为参数传给目标函数;
  • 如果被搬移函数调用了源函数中另一个函数,就必须把源对象传递给目标函数;如果需要源类的多个特性,也需将源对象传递给目标函数;如果目标函数需要太多源类特性,那最好分解目标函数,并将其中一部分移回源类。

Move Field (搬移字段)

  • 现象:在你的程序中,某个字段被其所驻类之外的另一个类更多地用到。所谓“使用”可能是通过设置、取值函数间接进行的。
  • 做法:在目标类新建一个字段,修改原字段的所有用户,令它们改用新字段
  • 也可能移动该字段的用户(某个函数),这取决于是否需要保持接口不受变化
  • 先使用Self Encapsulation Field (set/get)封装字段,好处:对值的修改不影响调用函数

Extract Class (提炼类)

  • 现象:某个类做了应该由两个类做的事情
  • 做法:建立一个新类,将相关的字段和函数从旧类搬移到新类
  • 一个类应该是一个清楚的抽象,处理一些明确的责任。但是,实际工作中,类会不断成长扩展,长时间下来,这个类会变得过分复杂。此时,你应该考虑哪些部分可以分离出去,并将它们分离到一个单独的类中。

Inline Class (将类内联化)

  • 现象:某个类没有做太多事情。如果一个类不再承担足够责任、不再有单独存在的理由(通常是因为此前的重构动作移走了这个类的责任),以lnline Class将其塞进另一个类中。
  • 做法:将这个类的所有特性(字段,方法)搬移到另一个类中,然后移除原类

Hide Delegate (隐藏委托关系)

  • 现象:客户通过一个委托类来调用另一个对象
  • 做法:在服务类上建立客户所需的所有函数,用以隐藏委托关系。
class Person {
	Department _department; //与Department类建立连接
	public Department getDepartment() {
		return _department;
	}
};

Class Department {
	private Person _manager;
	public Person getManager() {
		return _manager;
	}
};

//未隐藏调用关系
manager = john.getDepartment().getManager();
//隐藏调用关系
//在Person中建立委托函数
public Person getManager() {
	return _department.getManager();
}
manager = john.getManager();
//then, remove getDepartment() function from class Person.

Remove Middle Man (移除中间人)

  • 现象:某个类做了过多的简单委托动作
  • 做法:让客户直接调用受托类

Introduce Foreign Method (引入外加函数)

  • 现象:你需要为提供服务的类增加一个函数,但你无法修改这个类
  • 做法:在客户类中建立一个函数,并以第一参数形式传入 一个服务类实例
  • 外加函数终归是权宜之计

Introduce Local Extension (引入本地扩展)

  • 现象:你需要为服务类提供一些额外函数,但你无法修改这个类
  • 做法:建立一个新类,使它包含这些额外函数,让这个扩展品成为源类的子类或包装类
  • 尽量不要在子类中覆写原始类的函数,只添加新函数
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值