重构对象-Move Field搬移值域二
创始人
2025-05-29 12:44:04
0

重构对象-Move Field搬移值域二

1.搬移值域

1.1使用场景

如果我发现,对于一个field(值域),在其所驻class之外的另一个class中有更多函数使用了它,我就会考虑搬移这个field。
上述所谓「使用」可能是通过设值/取值(setting/getting)函数间接进行。我也可能移动该field的用户(某函数),这取决于是否需要保持接口不受变化。如果这些函数看上去很适合待在原地,我就选择搬移field。

1.2.如何做

  • 如果field的属性是public,首先使用Encapsulate Field封装值 将它封装起来。
  • 如果你有可能移动那些频繁访问该field的函数,或如果有许多函数访问某个field,先使用Self Encapsulate Field自动封装值 也许会有帮助
  • 编译,测试。
  • 在target class中建立与source field相同的field,并同时建立相应的设值/取值 (setting/getting)函数。
  • 编译target class。
  • 决定如何在source object中引用target object。
  • 一个现成的field或method可以助你得到target object。如果没有,就看能否轻易建立这样一个函数。如果还不行,就得在source class中新建一个field来存放target object。这可能是个永久性修改,但你也可以暂不公开它,因为后续重构可能会把这个新建field除掉。
  • 将所有「对source field的引用」替换为「对target适当函数的调用」。
  • 如果是「读取」该变量,就把「对source field的引用」替换为「对target取值函数(getter)的调用」;如果是「赋值」该变量,就把对source field的引用」替换成「对设值函数(setter)的调用」。
  • 如果source field不是private,就必须在source class的所有subclasses中查找source field的引用点,并进行相应替换。
  • 编译,测试。

1.3.示例

下面是Account class的部分代码

class Account...private AccountType _type;private double _interestRate;double interestForAmount_days (double amount, int days) {return _interestRate * amount * days / 365;}

我想把表示利率的_interestRate搬移到AccountType class去。目前已有数个函数引用了它,interestForAmount_days() 就是其一。
下一步我要在AccountType中建立_interestRate field以及相应的访问函数:

class AccountType...// 在新类中创建属性private double _interestRate;// 创建属性对应的get和set方法void setInterestRate (double arg) {_interestRate = arg;}double getInterestRate () {return _interestRate;}

这时候我可以编译新的AccountType class。
现在,我需要让Account class中访问此_interestRate field的函数转而使用AccountType对象,然后删除Account class中的_interestRate field。
我必须删除source field,才能保证其访问函数的确改变了操作对象,因为编译器会帮我指出未正确获得修改的函数。

  private double _interestRate;double interestForAmount_days (double amount, int days) {// 调用改为目标类中的属性return _type.getInterestRate() * amount * days / 365;}

使用Self Encapsulate(自我封装)
如果有很多函数已经使用了_interestRate field,我应该先运用Self Encapsulate Field

 class Account...private AccountType _type;private double _interestRate;double interestForAmount_days (double amount, int days) {return getInterestRate() * amount * days / 365;}private void setInterestRate (double arg) {// 多个位置使用了_interestRate变量_interestRate = arg;}private double getInterestRate () {// 多个位置使用了_interestRate变量return _interestRate;}

这样,在搬移field之后,我就只需要修改访问函数(accessors)就行了

   double interestForAmountAndDays (double amount, int days) {return getInterestRate() * amount * days / 365;}private void setInterestRate (double arg) {_type.setInterestRate(arg);}private double getInterestRate () {return _type.getInterestRate();}

如果以后有必要,我可以修改访问函数(accessors)的用户,让它们使用新对象。 Self Encapsulate Field 使我得以保持小步前进。如果我需要对做许多处理,保持小步前进是有帮助的。特别值得一提的是:首先使用Self Encapsulate Field 使我得以更轻松使用Move Method 将函数搬移到target class中。如果待移函数引用了field的访问函数(accessors),那么那些引用点是无须修改的。

相关内容

热门资讯

AWSECS:访问外部网络时出... 如果您在AWS ECS中部署了应用程序,并且该应用程序需要访问外部网络,但是无法正常访问,可能是因为...
AWSElasticBeans... 在Dockerfile中手动配置nginx反向代理。例如,在Dockerfile中添加以下代码:FR...
银河麒麟V10SP1高级服务器... 银河麒麟高级服务器操作系统简介: 银河麒麟高级服务器操作系统V10是针对企业级关键业务...
北信源内网安全管理卸载 北信源内网安全管理是一款网络安全管理软件,主要用于保护内网安全。在日常使用过程中,卸载该软件是一种常...
AWR报告解读 WORKLOAD REPOSITORY PDB report (PDB snapshots) AW...
AWS管理控制台菜单和权限 要在AWS管理控制台中创建菜单和权限,您可以使用AWS Identity and Access Ma...
​ToDesk 远程工具安装及... 目录 前言 ToDesk 优势 ToDesk 下载安装 ToDesk 功能展示 文件传输 设备链接 ...
群晖外网访问终极解决方法:IP... 写在前面的话 受够了群晖的quickconnet的小水管了,急需一个新的解决方法&#x...
不能访问光猫的的管理页面 光猫是现代家庭宽带网络的重要组成部分,它可以提供高速稳定的网络连接。但是,有时候我们会遇到不能访问光...
Azure构建流程(Power... 这可能是由于配置错误导致的问题。请检查构建流程任务中的“发布构建制品”步骤,确保正确配置了“Arti...