重构——简化条件表达式

1 Decompose Conditional(分解条件表达式)

从复杂表达式if-then-else三个段落中分别提炼出独立函数。
  Motivation:降低表达式逻辑复杂度,突出分支作用。
  做法:

  • 将if段落提炼出来,构成一个独立函数。
  • 将then段落和else段落都提炼出来,各自构成一个独立函数。

2 Consolidate Conditional Expression(合并条件表达式)

一系列条件表达式,得到的结果相同。将这些测试合并为一个条件表达式,并将这个条件表达式提炼成一个独立函数。
  Motivation:合并多个并列条件成一个明确的条件审查可以使检查用意更清楚;这项重构可以为Extract Method做准备。

  • 确定这些条件语句都没有副作用。
  • 使用恰当的逻辑操作符,将一系列相关条件表达式合并为一个。
  • 编译,测试。
  • 对合并后的条件表达式使用Extract Method。

3 Consolidate Duplicate Conditional Fragments(合并重复条件片段)

在条件表达式中有相同代码,将代码提取到条件表达式之外。

4 Remove Control Flag(移除控制标记)

在一系列布尔表达式中,某个变量带有控制标记作用,以break或return取代控制标记。
  Motivation:用break和continue语句的原因是用它们跳出复杂的条件语句可以使条件变得清晰得多。
  做法:

  • 找出让你跳出这段逻辑的判断标记。
  • 找出对标记变量赋值的语句,替代以恰当的break语句或continue语句。
  • 每次替换后,编译并测试。

5 Replace Nested Conditional with Guard Clauses(以卫语句取代嵌套条件表达式)

条件逻辑使人难以看清正常的执行路径。使用卫语句表现所有你特殊情况。
  Motivation:条件表达式一共有两种形式:1.所有分支都属于正常行为。2.条件表达式提供的答案中只有一种是正常行为,其他都不常见。如果分支都属于正常行为,使用if-else挺好的;如果某个条件极其罕见,就应该单独检查该条件,并在该条件为真时立即返回。这样单独的检查常常被称为“卫语句”。卫语句的精髓在于给一条分支以特别的重视。

  • 对于每个检查,放进一个卫语句。ps:卫语句要么就从函数中返回,要不就抛出一个异常。
  • 每次讲条件检查替换成卫语句后,编译并测试。

6 Replace Conditional with Polymorphism(以多态取代条件表达式)

有根据类型字段来选择函数不同的函数行为。将这个条件表达式的每个分支放进一个子类内的覆写函数中,然后将原始函数声明为抽象函数。
  Motivation:使用条件表达式,使用的是硬编码,很难对条件表达式进行变化。
  做法: 
  在做这个重构首发之前,必须得有继承结构,这种继承结构有两种手段去建立:Replace Type Code with Subclasses 和 Replace Type Code with State/Strategy。

  • 如果要处理的条件表达式是一个更大函数中的一部分,首先对条件表达式进行分析,然后使用extract Method把它提炼到一个独立函数中。
  • 如果有必要,使用Move Method将条件表达式放置到继承结构的顶端。
  • 任选一个子类,在其中建立一个函数,使之覆写超类中容纳条件表达式的那个函数。将与该子类先关的条件表达式分支赋值到新建函数中,并对它进行适当调整。
  • 编译,测试。
  • 在超类删除到表达式内被赋值的分支。
  • 编译,测试。
  • 重复上述的过程,知道所有分支都被移动到子类内。
  • 将超类之中容纳条件表达式的函数声明为抽象函数。

7 Introduce Null Object(引入Null对象)

你需要再三检查某对象是否为null。将null值替换为null对象。
  Motivation:这是践行多态的一个非常好的方案。试想,如果对每个对象,都必须判断其是否为null,然后才能进行下一步的行为。那么系统中一定会运行大量的判断代码,这是就大量的重复代码。
  做法:

  • 为源类建立一个子类,使其行为就像源类的null版本。在源类和null子类中都加上isNull(),前者返回false,后者返回true。
  • 编译。
  • 找出所有索求源对象却获得一个null的地方。修改这些地方,使它们调用isNull()函数。
  • 编译,测试。
  • 找出这样的程序点:如果对象不是null,做A动作,否则B动作。
  • 对于每一个上述弟弟昂,在null类中覆盖A动作,使其行为和B相同。
  • 使用上述被覆写的动作,然后删除对象是否为null的条件测试。编译并测试。

tips:其实这样的方式可以被认为是特例类模式,用特例来减少处理特例的开销。

8 Introduce Assertion

用断言明确表明对程序状态的假设。
  Motivation:可以帮助我们理解某些条件对程序是极其重要的。断言可以帮助程序阅读者阅读代码,可以帮助调试程序。
  做法:
  只要程序源不犯错,断言就应该不会对系统运行造成任何影响,所以加入断言永远不会影响程序的行为。断言一定要用来判断一定为真的条件。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容

  • 1 Rename Method(函数改名) 修改函数名字以良好表达函数的用途。Motivation:各种大师特别推...
    hklbird阅读 515评论 0 0
  • 简化条件表达式 9.1 Decompose Conditional (分解条件表达式) 你有一个复杂的条件表达式语...
    rxdxxxx阅读 488评论 0 0
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,608评论 18 399
  • 洁净光芒 墙上白色无暇 自影子 呼吸和黑夜来了 墙有了年龄般在积累长大 楼梯一层一层 树叶一叠一叠 沉重和腐化 我...
    零温度阅读 169评论 0 3
  • 有时候 我看见你眼里的忧伤 你总把我当任性的孩子 有时候 我看见你的脚步匆匆忙忙 你说那只是一种习惯 有时候 我看...
    DinoF阅读 231评论 0 1