返回总目录


本小节目录:

1、Extract Method(提炼函数)

2、Inline Method(内联函数)

3、Inline Temp(内联临时变量)


1 Extract Method(提炼函数)

概要

你有一段代码可以被组织在一起并独立起来。

将这段代码放进一个独立函数中,并让函数名称解释该函数的作用。

动机

如果有一个过长的函数或者需要一段注释才能让人理解用途的代码,那么就将这段代码放进一个独立函数中。

简短而且命名良好的函数的好处:

  • 函数粒度小,被复用的机会大
  • 使高层函数读起来像注释
  • 函数都是细粒度,更易被复写

还记得在第二种坏味道——Long Method中说的函数到底有多长吗?作者如是说:“在我看来,长度不是问题,关键在于函数名称和函数本体之间的语义距离。如果提炼可以强化代码的清晰度,那就去做,就算函数名称比提炼出来的代码还长也无所谓。”

范例

    public class Receipt
{
private List<decimal> Discounts { get; set; }
private List<decimal> ItemTotals { get; set; } public decimal CalculateGrandTotal()
{
decimal subTotal = 0m;
//计算subTotal 的总和
foreach (decimal itemTotal in ItemTotals)
{
subTotal += itemTotal;
}
//subTotal 要循环减去discount,也就是计算Discount
if (Discounts.Count > )
{
foreach (decimal discount in Discounts)
{
subTotal -= discount;
}
}
//计算Tax
decimal tax = subTotal * 0.065m;
subTotal += tax;
return subTotal;
} }

重构后代码如下:

 public class Receipt
{
private List<decimal> Discounts { get; set; }
private List<decimal> ItemTotals { get; set; } public decimal CalculateGrandTotal()
{
decimal subTotal = CalculateSubTotal(); subTotal = CalculateDiscounts(subTotal); subTotal = CalculateTax(subTotal); return subTotal;
}
//计算subTotal 的总和
private decimal CalculateSubTotal()
{
decimal subTotal = 0m;
foreach (decimal itemTotal in ItemTotals)
{
subTotal += itemTotal;
}
return subTotal;
}
//计算折扣
private decimal CalculateDiscounts(decimal subTotal)
{
if (Discounts.Count > )
{
foreach (decimal discount in Discounts)
{
subTotal -= discount;
}
}
return subTotal;
}
//计算Tax
private decimal CalculateTax(decimal subTotal)
{
decimal tax = subTotal * 0.065m;
subTotal += tax;
return subTotal;
} }

小结

这个重构手法是不是很简单。我相信这个手法大多数人用的非常多,或许你潜意识里就是这么做的。

我不知道大家的公司有没有在代码编写规范里面把这个作为参考,比如一个方法最多不能超过多少行等等。这在一定程度上也能使程序员把这些复杂的逻辑剥离成意义很清楚的小方法。

2 Inline Method(内联函数)

概要

一个函数的本体与名称同样清楚易懂。

在函数调用点插入函数主体,然后移除该函数。

动机

使用间接层可以让函数通俗易懂,但是如果这个函数本体本就通俗易懂,那这个间接层就是无用的间接层,可以将其去除。

范例

int GetRating()
{
return MoreThanFiveLateDeliveries() ? : ;
} bool MoreThanFiveLateDeliveries()
{
return _numberOfLateDeliveries > ;
}

对于这个函数来说,MoreThanFiveLateDeliveries这个方法就是一个无用的间接层。因为原函数本就很清晰,此处将其去除。

重构后代码如下:

int GetRating()
{
return _numberOfLateDeliveries > ? : ;
}

小结

这个手法是比较简单的。但是不要因为简单就不注意它。在内联函数的时候,要确定该函数不具有多态性,因为子类无法覆写一个根本不存在的函数。

3 Inline Temp(内联临时变量)

概要

你有一个临时变量,只被一个简单表达式赋值一次,而它妨碍了其他重构手法。

将所有对该变量的引用动作,替换为对它赋值的那个表达式自身。

动机

这个手法多半是作为下一个要讲的手法Replace Temp with Query的一部分来使用的。

单独使用的情况是:某个临时变量被赋予某个函数调用的返回值,而这个临时变量妨碍了其他重构手法。

范例

bool GetBasePrice()
{
double basePrice = anOrder.GetBasePrice();
return basePrice > ;
}

重构后代码如下:

bool GetBasePrice()
{
return anOrder.GetBasePrice() > ;
}

小结

在将临时变量替换为表达式自身的时候要注意,这个临时变量在该函数内是否只被赋值一次。如果不是一次,那么就不要这么做。

To Be Continued...

重构手法之Extrct Method(提炼函数)的更多相关文章

  1. 重构改善既有代码设计--重构手法01:Extract Method (提炼函数)

    背景: 你有一段代码可以被组织在一起并独立出来.将这段代码放进一个独立函数,并让函数名称解释该函数的用途. void PrintOwing(double amount) { PrintBanner() ...

  2. 重构改善既有代码设计--重构手法08:Replace Method with Method Object (以函数对象取代函数)

    你有一个大型函数,其中对局部变量的使用,使你无法釆用 Extract Method. 将这个函数放进一个单独对象中,如此一来局部变量就成了对象内的值域(field) 然后你可以在同一个对象中将这个大型 ...

  3. 重构改善既有代码设计--重构手法16:Introduce Foreign Method (引入外加函数)&& 重构手法17:Introduce Local Extension (引入本地扩展)

    重构手法16:Introduce Foreign Method (引入外加函数)你需要为提供服务的类增加一个函数,但你无法修改这个类.在客户类中建立一个函数,并以第一参数形式传入一个服务类实例. 动机 ...

  4. 重构改善既有代码设计--重构手法10:Move Method (搬移函数)

    你的程序中,有个函数与其所驻类之外的另一个类进行更多的交流:调用后者,或被后者调用.在该函数最常用引用的类中建立一个有着类似行为的新函数.将旧函数编程一个单纯的委托函数,或是将旧函数完全移除. 动机: ...

  5. 重构手法之Replace Temp with Query(以查询取代临时变量)

    返回总目录 6.4Replace Temp with Query(以查询取代临时变量) 概要 你的程序以一个临时变量保存某一表达式的运算结果. 将这个表达式提炼到一个独立函数中.将这个临时变量的所有引 ...

  6. 重构手法之Introduce Explaining Variable(引用解释性变量)

    返回总目录 6.5Introduce Explaining Variable(引用解释性变量) 概要 你有一个复杂的表达式. 将该复杂表达式(或其中一部分)的结果放进一个临时变量,以此变量名称来解释表 ...

  7. 重构改善既有代码设计--重构手法05:Introduce Explaining Variable (引入解释性变量)

      发现:你有一个复杂的表达式. 解决:将该复杂的表达式(或其中的部分)的结果放进一个临时变量,并以此变量名称来解释表达式用途. //重构前 if((platform.toUpperCase().in ...

  8. 重构改善既有代码设计--重构手法04:Replace Temp with Query (以查询取代临时变量)

    所谓的以查询取代临时变量:就是当你的程序以一个临时变量保存某一个表达式的运算效果.将这个表达式提炼到一个独立函数中.将这个临时变量的所有引用点替换为对新函数的调用.此后,新函数就可以被其他函数调用. ...

  9. 重构手法之Split Temporary Variable(分解临时变量)

    返回总目录 本小节目录 Split Temporary Variable(分解临时变量) Remove Assignments to Parameters(移除对参数的赋值) 6.6Split Tem ...

随机推荐

  1. JS表单提交的几种方式

    第一种方式 : 表单提交,在 form 标签中增加 onsubmit 事件来判断表单是否提交成功 <script type="text/javascript"> fun ...

  2. File FileStream StreamWriter StreamReader文件读写操作方法

    string path = "D:\\AccountChecking\\Test.txt"; string content = "abcdefg\r\nhigklmn\r ...

  3. Python系列之多线程、多进程

    线程是操作系统直接支持的执行单元,因此,高级语言通常都内置多线程的支持,Python也不例外,并且,Python的线程是真正的Posix Thread,而不是模拟出来的线程. Python的标准库提供 ...

  4. Python内置类型(2)——布尔运算

    python中bool运算符按优先级顺序分别有or.and.not, 其中or.and为短路运算符 not先对表达式进行真值测试后再取反 not运算符值只有1个表达式,not先对表达式进行真值测试后再 ...

  5. gitlab仓库迁移

    遇到一个情况,需要将两个gitlab仓库合并.好在都是使用的ldap账户登陆,用户账户不需要迁移. 实际的使用情况下,需要迁移的主要部分为分组及分组下项目.gitlab的api还是很给力的,能够获取所 ...

  6. web自动化测试从入门到持续集成(selenium webdriver)

    在很多刚学习自动化的可能会认为我只需要会运用selenium,我只需要在一个编辑器中实用selenium +java编写了一些脚本那么就会自动化了,是真的吗?答案肯定是假的.自动化肯定是需要做到真的完 ...

  7. Android 性能优化概念(1)

    http://www.open-open.com/lib/view/open1421723359718.html#_label0 阅读目录 0)Render Performance 1)Underst ...

  8. Mvc 流程调用分析

    链接地址 https://www.processon.com/view/link/59e71fbbe4b09000f03ce78e 总结: 1. 在Global.ascx 中我们使用RouteColl ...

  9. 如何学习LoadRunner性能测试?

    最近组内同事针对性能测试LR的脚本部分做了介绍,是个不错的分享.会后反思自己也有很长一段时间没做性能测试了,根据以往的经验,有必要做些整理和补充,本文主要介绍一些Loadrunner性能测试的学习方法 ...

  10. Tomcat 笔记-设置虚拟主机

    通过作用虚拟主机,可以使多个不同域名的网站共存于一个Tomcat中 在tomcat的server.xml文件中添加主机名: <Host name="hostname" app ...