编写可维护软件的10大要则-低层级原则

本篇主要总结《代码不朽 编写可维护软件的10大要则》中的低层级原则,包括如下内容:

一. 编写短小的代码单元

1. 原则:

  • 代码单元的长度应该限制在15行代码以内
  • 你应该编写不超过15行代码的单元,或者将长的单元分解成多个更短的单元,直到每个单元都不超过15行代码。
  • 该原则能提高可维护性的原因在于,短小的代码单元易于理解、测试及重用。

2. 动机:

短小的代码单元易于测试、分析、重用

  • 短小的代码单元易于测试
    一般来说,短小的代码单元可能只做一件事,而较长的代码单元会尝试做多件事,拥有单一职责的代码更易于测试。
  • 短小的代码单元易于分析
    相比代码行数较多的代码单元,分析短小单元内部原理所花费时间更少。
  • 短小的代码单元易于重用
    较长的代码单元试图提供各种具体的实现细节,或者几个功能的固定组合,因此提供的功能范围也更加特定。短小的代码单元职责相对单一,更易于重用。

3. 如何使用本原则:

  • 场景一:编写一个新的代码单元时
    编写一个新的代码单元时,绝对不要让它超过15行代码。当你写到第15行代码时,你需要开始思考如何添加下一步的功能。它真的属于你当前编写的单元,还是应该有自己的代码单元?如果单元中的代码确实超过15行,需要想办法将它变得更短。
  • 场景二:向代码单元中添加新功能时
    当你向系统添加新功能时,代码单元开始变得越来越长,需要坚持原则,绝不能超过代码行数的限制。如果超过限制,需要考虑重构代码。

使用重构来实现该原则:
1. 提取方法
2. 将方法替换为方法对象

4. 本节内容总结:

有人可能会质疑代码单元的长度限制在15行以内是否可以实现需求功能。根据SIG的代码检测分析,优秀的项目1-15行代码占比甚至会高于90%。
实际上,要求项目中每个代码单元都不超过15行也不太现实,它只是我们编写代码单元时的目标。每当超过15行时,我们要停下来思考是否必须要超过15行,有何办法可以让代码单元不超过15行。你可以先完整地实现代码单元的功能,然后再停下来思考该如何重构以缩短其长度。如此以来,不仅可以提高代码的可维护性,更能大幅提高个人的编码水平。


二. 编写简单的代码单元

1. 原则:

  • 限制每个代码单元分支点的数量不超过4个
  • 应该将复杂的代码单元拆分成多个更简单的单元,避免多个复杂的单元在一起。
  • 该原则能提高可维护性的原因在于,分支点越少,代码单元越容易被修改和测试。

2. 动机:

简单的代码单元易于修改、测试。

  • 简单的代码单元易于修改
    高复杂度的代码单元通常难以理解,更难以修改。
  • 简单的代码单元易于测试
    代码单元的分支点越多,要覆盖所有分支的测试用例就越多,难以测试。

3. 如何使用本原则:

  • 场景一:代码单元本身包含较多会增加分支点的操作符
    代码单元包含较多会增加分支点的操作符,如if、case、&&、||、while、for、for...each、catch等。这说明代码单元逻辑较复杂,需要通过提取方法来降低复杂度。

  • 场景二:处理链式条件语句
    对于链式的if-then-else语句,或者switch-case语句,每个条件都是互斥的,可以通过引入更合适的数据结构引入多态来降低复杂度。

  • 场景三:处理嵌套条件语句
    对于嵌套层级多的条件语句,可以通过卫语句 & 提取方法来降低复杂度。

4. 本节内容总结:

代码单元简单意味着代码单元分支点少,最简单的情况就是整个代码单元顺序执行,没有多余分支。每多一个分支,理解起来相当于要多“压栈”一次,编写测试代码时就需要增加一个测试用例。
圈复杂度是代码复杂性的衡量标准,理解了圈复杂度,就理解了编写简单代码的必要性。


三. 不写重复代码

1. 原则:

  • 不要复制代码。
  • 你应该编写可重用的、通用的代码,或者调用已有的代码。
  • 该原则能提高可维护性的原因在于,如果复制代码,就需要在多个地方修复bug,不仅低效,且容易出错。

重复代码的类型:

  • 1类克隆:
    含有至少6行相同代码(不包括空行和注释行)。
  • 2类克隆:
    在语法上相同的两段代码,如下述两段代码:
public void setPageWidthInInches(float newWidth) {
    float cmPerInch = 2.54f;
    this.pageWidthInCm = newWidth * cmPerInch;
    ...
}

public void setPageWidthInPoints(float newWidth) {
    float cmPerPoint = 0.03527f;
    this.pageWidthInCm = newWidth * cmPerPoint;
    //其他代码,同setPageWidthInInches中的一样
}

2. 动机:

  • 重复代码更加难以分析
    倘若问题出在有重复代码的地方,必须查找并分析所有重复的地方,增加了处理难度。

  • 代码重复更加难以修改
    重复代码中的bug会在系统中出现多次,需要修改多次才行。

3. 如何使用本原则:

  • 提取方法
  • 提取父类

具体细节可参考《重构 改善既有代码的设计》

4. 本节内容总结:

重复代码是“万恶之源”,是最大的软件质量问题,要切记不要在系统中出现重复代码。


四. 保持代码单元的接口简单

1. 原则:

  • 限制每个代码单元的参数不能超过4个。
  • 你应该将多个参数提取成对象。
  • 该原则能提高可维护性的原因在于,较少的参数可以让代码单元更容易理解和重用。

2. 动机:

保持较少的接口参数并引入合适的参数对象,能够带来很多的好处。接口参数较少的方法能够保持简单的上下文,因此更容易被人理解。除此以外,由于它们并不过于依赖来自外部的输入,所以也更易于重用和修改。

  • 短接口更易于理解和重用
    随着代码库规模的增长,核心类会逐渐成为其他代码所依赖的API。为了避免代码量的迅速膨胀以及开发速度下降,核心类必须有一个清晰、短的接口定义。

  • 短接口的方法更易于修改
    过长的接口不仅让方法变得难以理解,许多情况下还表示它承担着多重的职责。从这个角度上讲,接口长短与代码单元大小和复杂度是有关系的,即接口过长的方法更难以修改。

3. 如何使用本原则:

  • 改进数据模型
    过长的接口本身不是问题,而是实际问题的一个表现--意味着代码中可能存在着设计不合理的数据模型,或者随意的改动。可以将接口长短视为一种代码坏味道,通过它来了解是否需要改进你的数据模型。
  • 使用方法对象替换方法

4. 本节内容总结:

根据实践中的经验,将参数个数的上限设为4个是比较合理的。5个参数的方法已经开始变得难以理解,并且担负过多的职责。


五. 编写简洁的代码

1. 原则:

  • 编写简洁的代码
  • 你不应该在完成开发工作后留下代码坏味道。

2. 动机:

简洁的代码是可维护的代码。

3. 如何使用本原则

SIG从长期的咨询经验中总结出如下7条规则来避免产生最不利于可维护性的代码坏味道:

  • 不要编写单元级别的代码坏味道
    此处的单元基本指的是函数(方法)级别,要遵守这条规则,需要及时重构“坏味道”代码。

  • 不要编写不好的注释
    注释有些时候被看作是代码不好的表现。从实际经验中可以确认,行内注释通常意味着工程方案不够优雅。
    最好的选择就是使用好的命名使得代码可以自注释。

  • 不要注释代码
    不要提交被注释掉的代码,版本控制系统会保留旧代码的记录,你可以安全地删除它们。
    从原始开发人员的角度,FIXME注释及相关代码是能够理解的,但是对于新的开发人员却是一种干扰。

  • 不要保留废弃代码
    废弃代码指的是根本不会被执行或者输出被“废弃”的代码。
    废弃代码的几种形式:
    a. 注释掉的代码
    b. 方法中无法到达的代码
    c. 无用的私有方法
    d. 注释中的代码

  • 不要使用过长的标识符名称
    通常要避免使用过长的标识符名称,团队应该有一个正式的命名规范。

  • 不要使用魔术常量
    魔术常量指的是在代码中没有被清晰定义的数字或者字符串。

  • 不要使用未正确处理的异常
    3个正确处理异常的原则:
    a. 捕获一切异常
    记录下系统的失败行为,是为了了解它们产生的原因并加以改进。空的catch代码块也能够编译通过,但这是一个不好的实践行为,它没有提供任何与异常上下文有关的信息。
    b. 捕获特定异常
    为了可以追踪某些特定事件的异常,应该捕获特定的异常。不应该直接捕获Throwable、Exception、RuntimeException异常。
    c. 在展示给终端用户之前
    先将特定的异常信息转换成通用的信息。用户不应该被具体的异常信息所“打扰”,因为这会让他们感到困惑,而且也会带来安全隐患(例如提供了有关系统内部工作原理的过多信息)。

4. 本节内容总结:

实际工作中会使用更多的编码规范和质量检查规范,这7个规范仅仅是SIG团队认为对编写可维护代码最重要的几条,应该严格遵守。

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

推荐阅读更多精彩内容