重构之重新组织函数

重构手法中,很大一部分是对函数进行整理,使之更恰当的包装代码。几乎所有时刻,问题都源于过长函数。对于过长函数,一项重要的重构手法就是提炼函数,它把一段代码从原先函数中提取出来,放进一个单独函数中。内联函数正好相反:将一个函数调用替换为该函数本体。如果在多次提炼之后,意识到提炼所得的某些函数并没有做任何实质事情,或如果需要回溯回复原先函数,就需要内联函数。
一.提炼函数
提炼函数是最常用的重构手法之一。当看见一个过长的函数或者一段需要注释才能让人理解用途的代码,我就会将这段代码放进一个独立函数中。提炼函数的好处有很多,首先,如果每个函数的粒度都很小,那么函数被复用的机会就更大;其次,这会使高层函数读起来就像一系列注释;再次,如果函数都是细粒度,那么函数的覆写也会更容易些。
做法:
(1)创造一个新函数,根据这个函数的意图来对它命名(以它"做什么"来命名,而不是以它"怎样做"命名)。
(2)将提炼出的代码从源函数复制到新建的目标函数中。
(3)仔细检查提炼出的代码,看看其中是否引用了"作用域限于源函数"的变量。
(4)检查是否有"仅用于被提炼代码段"的临时变量。如果有,在目标函数中将它们声明为临时变量。
(5)检查被提炼代码段,看看是否有任何局部变量的值被它改变。如果一个临时变量值被修改了,看看是否可以将被提炼代码段处理为一个查询,并将结果赋值给相关变量。如果很难这样做,或如果被修改的变量不止一个,你就不能仅仅将这段代码原封不动地提炼出来。你可能需要分解临时变量,然后再尝试提炼。
(6)将被提炼代码段中需要读取的局部变量作为参数传给目标函数。
二.内联函数
我们经常以简单的函数表现动作意图,这样会使代码更清晰易读。但有时候你会遇到某些函数,其内部代码和函数名称同样清晰易读。果真如此,你就应该去掉这个函数,直接使用其中的代码。间接性可能带来帮助,但非必要的间接性总是让人不舒服。另一种需要使用内联函数的情况是,你手上有一群组织不甚合理的函数。你可以将它们都内联到一个大型函数中,再从中提炼出组织合理的小型函数。这么看的话内联函数和提炼函数是完全相反的,什么时候我们需要提炼函数,什么时候内联函数,需要我们自己把握。
三.内联临时变量
你有一个临时变量,只被一个简单表达式赋值一次,而它妨碍了其他重构手法。内联临时变量多半是作为以查询取代临时变量的一部分使用的。唯一单独使用内联临时变量的情况是:你发现某个临时变量被赋予某个函数调用的返回值,一般来说这样的变量不会有任何危害,可以放心地把它留在那儿。但如果这个临时变量妨碍了其他的重构手法,就应该将它内敛化。简单来说,没有必要的临时变量就把它去掉。
四.以查询取代临时变量
你的程序以一个临时变量保存某一表达式的运算结果,将这个表达式提炼到一个独立函数中,将这个临时变量的所有引用点替换为对新函数的调用,此后,新函数就可被其他函数使用。
临时变量的问题在于:它们是暂时的,而且只能在所属函数内使用。由于临时变量只在所属函数内可见,所以他们会驱使你写出更长的函数,因为只有这样你才能访问到需要的临时变量。如果把临时变量替换为一个查询,那么同一类中的所有函数都可以获得这份信息。这个重构手法较为简单的情况是:临时变量只被赋值一次,或者赋值给临时变量的表达式不瘦其他条件影响。
五.引入解释性变量
表达式有可能非常复杂而难以阅读,这种情况下,临时变量可以帮助你将表达式分解为 比较容易管理的形式(引入解释型变量和前面的方法也是相反的)。引入解释性变量时一个很常见的重构手法,不过我几乎总是尽量使用提炼函数来解释一段代码的意义,毕竟临时变量只在它所处的那个函数中才有意义,局限性较大。
六.分解临时变量
临时变量有各种不同用途,其中某些用途会很自然地导致临时变量被多次赋值。临时变量应该只被赋值一次,如果它们被赋值超过一次,就意味着它们在函数中承担了一个以上的责任。如果临时变量承担多个责任,就应该呗替换为多个临时变量,每个变量只承担一个责任。同一个临时变量承担两件不同的事情,会让代码阅读者糊涂。
七.移除对参数的赋值
对参数赋值的意思是:如果你把一个名为foo的对象作为参数传给某个函数,那么"对参数的赋值"意味改变foo,使它引用另一个对象。如果你在"被传入对象"身上进行什么操作,那没问题。这里只很对"foo呗改为指向另一个对象"这种情况讨论。这种情况降低了代码的清晰度,而且混用了按值传递和按引用传递这两种参数的传递方式,Java只采用按值传递方式。在按值传递的情况下,对参数的任何修改,都不会对调用端造成任何影响。
八.以函数对象取代函数
你有一个大型函数,其中对局部变量的使用使你无法使用提炼函数。将这个函数放进一个单独对象中,如此一来局部变量就成了对象内的字段。然后你可以在同一个对象中将这个大型函数分解为多个小型函数。
如果一个函数之中局部变量泛滥成灾,那么像分解这个函数是非常困难的。以函数对象取代函数可以助你减轻这一负担,以函数取代对象会将所有局部变量都编程函数对象的字段,然后就可以用这个新对象使用提炼函数创造出新函数,从而将原本的大型函数拆解变短。
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值