代码是如何腐化的

标签: 代码 | 发表时间:2013-07-05 14:26 | 作者:
出处:http://agiledon.github.com/

代码是如何腐化的?这是一个很大的话题,因为这种腐化的代码样本可能会体现不同的特征,若要彻底总结,可能会又是一本《重构》。我自然没有这个能力和知识。好在有一个简便的说法,即可以诉诸于“破窗理论”的威力。无论多少坏味道识别,重构手法运用,提高代码质量的最佳实践,以及运用诸多甄别代码质量体征的工具,都仅仅限于“术”的运用而已。若未能在开发人员内心树立整洁代码的习惯,时时刻刻对各种代码臭味保持敏感,且具有一颗期待卓越代码之心,那么,随着项目的演进,时间的推移,代码最终还是会慢慢腐烂。

这几日在开发一个User Story时,从诸多测试代码(包括集成测试与验收测试)中,观察到了一些接近腐烂的代码坏味。这些代码虽然不是产品代码,但同样是我们交付工件的一部分。最关键之处在于:它让我察觉到一种危险的趋势,若不能及时扭转,可能会让代码陷入腐烂的泥沼。若能及时解决这些糟糕代码,其实仅仅需要一些简单的重构手法,付出几个小时时间即可。

首先是针对集成测试的数据准备。我们要编写的集成测试针对Spring Batch Job,这些Job需要访问数据库,以验证Job的执行是否符合期望。我们发现在之前已有与Spring Batch Job相关的集成测试存在,并提供了访问数据库,以及启动、访问和停止Ftp服务器的功能。其中,与数据准备有关的功能放到单独定义的Fixture类中。这些Fixture是为特定目的编写的数据准备,可是,随着越来越多的Batch Job出现,有诸多集成测试都需要准备数据,于是开始慢慢产生了测试数据的重叠,逐步浮现出违背DRY原则的征兆了。

对于多数程序员而言,并非不重视重用,但多数却不愿意为了重用付出一些代价。例如针对一些具备差异性的功能,一些程序员更愿意使用Copy And Paste,然后再针对自己的需求对实现进行修改或调整。观察目前的一些集成测试,正是这样一些陋习导致的。

在这些集成测试中,使用了继承的方式来重用数据准备的功能。如下图所示:

在CustomerIntegratedDataFixture中,提供了相关方法实现了对Customer数据的创建。由于需要提供访问FtpServer的功能,因此又定义了CustomerIntegratedDataAndFtpPrepareFixture类,使其继承CustomerIntegratedDataFixture。它定义了startFtpServer()和stopFtpServer()方法,并在JUnit中,运用了@BeforeClass与@AfterClass标记,使其避免为每个测试启动和停止专有的FtpServer。现在,我们编写的集成测试同样需要与Customer有关的数据,但并不需要Ftp功能。换言之,我们希望重用CustomerIntegratedDataFixture。现在看来,似乎并没有问题。例如,我们可以让新增的测试直接继承CustomerIntegratedDataFixture。然而,就在同样的集成测试模块中,我们还发现了其他集成测试同样编写了自己的数据准备类。这些数据准备与Spring Batch Job无关,却同样提供了准备Customer数据的功能。存在的差异是它除了提供Customer数据外,还提供了依赖Customer的Consent数据。

我们没有着急去重用CustomerIntegratedDataFixture,因为我们察觉到代码会随着这种继承体系的延伸,会变得越来越难以重用。如上图的继承体系,使得数据准备与Spring Batch Job紧耦合了,同时又在CustomerIntegratedDataAndFtpPrepareFixture子类中引入了与Ftp有关的耦合,明显违背了单一职责原则。我们需要单独剥离出数据准备的类,它即可以作为超类被集成测试类继承,也可以通过组合的方式被继承了JobLauncherTestUtils的测试子类所调用。这符合Bridge模式的设计原则。因此,我们运用了“Replace Inheritance with Delegation”手法,对其进行了简单重构:

之后,我们对Customer和Consent对应的数据准备类进行了相应的重构与修改,使得这些数据的准备更为内聚,并去除一些不必要的重复,使之更容易被重用。

第二个例子是在JBehave的Story中,我看到了这样的Steps类的组织,如图:

我们看到了什么?——一个“扁平组织”的Steps类。显然,促成这样的结果是一个渐进的过程。由于在之前编写相关的Steps类时,还看不到分类的概念,因此,只是简单地将自己的Steps类放到step之下即可。然后,不断有开发人员增加自己的Steps类,他们找到了step位置,却没有仔细思考是否需要更好地对Steps类进行组织。这就使得Steps类略显零乱,没有展现出好的结构。我们重新组织了这些Steps类:

只需要简单地归类,调整结构,整个Steps类就变得更加清晰了。于是,我们发现了可以重用的可能。观察重新组织之后的batch包,这里面包含的UpdateCustomerTypeSteps,ProductSystemLinkLoaderSteps与DeleteOrphanedRecordsSteps,都是与Btach Job有关的Steps类。MaintainProspectsSteps类则是我们新增的类,它同样需要用到启动Batch Job的方法。在之前存在的Steps类中,已经存在相似的代码了。例如在UpdateCustomerTypeSteps类中:

          private String waitAndGetSatus(Map<String, String> params) throws InterruptedException {
        String status = null;
        for (int index = 0; index < MAX_TRY_TIME; index++) {
            status = getJobStatus(params);
            if ("COMPLETED".equals(status)) {
                break;
            }
            Thread.sleep(WAIT_INTERVAL);
        }
        return status;
    }

    private String getJobStatus(Map<String, String> params) {
        return db2JdbcTemplate.queryForObject("select jobExec.STATUS from TCIST_JOB_EXECUTION jobExec " +
                "inner join TCIST_JOB_PARAMS jobParam on jobExec.JOB_INSTANCE_ID = jobParam.JOB_INSTANCE_ID " +
                "and jobParam.KEY_NAME = 'retry' " +
                "and jobParam.STRING_VAL = ?", String.class, params.get("retry"));
    }

再看DeleteOrphanedRecordsSteps类:

          private String waitAndGetSatus(String currentTime) throws InterruptedException {
        String status = null;
        for (int index = 0; index < MAX_TRY_TIME; index++) {
            status = getJobStatus(currentTime);
            if ("COMPLETED".equals(status)) {
                break;
            }
            Thread.sleep(WAIT_INTERVAL);
        }
        return status;
    }

    private String getJobStatus(String currentTime) {
        return jdbcTemplate.queryForObject("select jobExec.STATUS from TCIST_JOB_EXECUTION jobExec " +
                "inner join TCIST_JOB_PARAMS jobParam on jobExec.JOB_INSTANCE_ID = jobParam.JOB_INSTANCE_ID " +
                "and jobParam.KEY_NAME = 'time' " +
                "and jobParam.STRING_VAL = ?", String.class, currentTime);
    }

比较这些方法,除了jobParam的key与value存在细微区别,其余实现完全相同。若按照这样一个态势发展,随着与Batch Job有关的Story逐渐增多,不发现这种代码的臭味并即刻解决,这些代码就会逐渐蔓延,最后变得“无法自拔”。想要修改,已经变得极为困难了。

我们为这些Steps类提供了一个抽象的超类AbstractBatchJobSteps,并将这些可能重用的方法提取到这个超类中:

      public class AbstractBatchJobSteps extends AbstractSteps {
    private static final int WAIT_INTERVAL = 1000;
    private static final int MAX_TRY_TIME = 30;

    protected String waitAndGetSatus(Map<String, String> params, String paraKey) throws InterruptedException {
        String status = null;
        for (int index = 0; index < MAX_TRY_TIME; index++) {
            status = getJobStatus(params, paraKey);
            if ("COMPLETED".equals(status)) {
                break;
            }
            Thread.sleep(WAIT_INTERVAL);
        }
        return status;
    }

    private String getJobStatus(Map<String, String> params, String paraKey) {
        return jdbcTemplate.queryForObject("select jobExec.STATUS from TCIST_JOB_EXECUTION jobExec " +
                "inner join TCIST_JOB_PARAMS jobParam on jobExec.JOB_INSTANCE_ID = jobParam.JOB_INSTANCE_ID " +
                "and jobParam.KEY_NAME = '" + paraKey + "' " +
                "and jobParam.STRING_VAL = ?", String.class, params.get(paraKey));
    }
}

如上的例子都可以通过一些细小的重构手法改进代码,使得代码的结构更加清晰,并有利于代码的重用。我深信大多数开发人员都具备这样的技能,且只需要稍加思索,即能发现这些代码的坏味。然而,我们总是因为种种原因,对这种还不太严重的“破窗”风景视而不见。殊不知当我们开始对这种不够整洁的代码采取纵容态度时,就可能会是代码腐化之始。一旦真正腐化,就将积重难返,到了那时,我们就可能真正无能为力了。

你是否遭遇过这样的情形?面对一个承担了无数职责似乎无所不能的上帝类,它被无数多的Client调用,且又没有足够覆盖率的测试,你是否会产生心有余而力不足的感慨。这时的你,是否像一位奋战沙场,出生入死却无力挽回败局的将军,面对那汹涌而来占据压倒性优势的敌军,唯有对天长叹:“某有心杀贼,却无力回天啊!”

相关 [代码] 推荐:

代码重构

- - ITeye博客
随着程序的演化,我们有必要重新思考早先的决策,并重写部分代码. 代码需要演化;它不是静态的事物. 重写、重做和重新架构代码合起来,称为重构.    当你遇到绊脚石  ---  代码不在合适,你注意到有两样东西其实应该合并或是其他任何对你来说是"错误"的东西  -------- . 如果代码具备以下特征,你都应该考虑重构代码:.

代码小比较

- Tim - 斯巴达第二季
判断上百万个4k的buffer是否为全0,我最先想到的办法是:zero_buffer = malloc(4096);. /* 循环百万次读取buffer */.         /* 全0 */. 由于好奇,看看shell工具cp的代码,它的解决办法是:. /* 循环百万次读取buffer */.         /* 全0 */.

两行 JavaScript 代码

- MessyCS - Dreamer&#39;s Blog
最近看到了两行 JavaScript 代码,很受启发. 在 JavaScript 中,我们可以获取HTML元素的属性值,例如 element.id. 但是,因为 for 和 class 是 JavaScript 中的关键字,所以在 JavaScript 中这两个属性名称分别用 htmlFor 和 className 代替,于是在封装的时候需要先对这两个属性进行特殊判断.

Netty代码分析

- LightingMan - 淘宝JAVA中间件团队博客
Netty提供异步的、事件驱动的网络应用程序框架和工具,用以快速开发高性能、高可靠性的网络服务器和客户端程序[官方定义],整体来看其包含了以下内容:1.提供了丰富的协议编解码支持,2.实现自有的buffer系统,减少复制所带来的消耗,3.整套channel的实现,4.基于事件的过程流转以及完整的网络事件响应与扩展,5.丰富的example.

python代码调试

- - 阿里古古
【转自: http://blog.csdn.net/luckeryin/article/details/4477233】. 本文讨论在没有方便的IDE工具可用的情况下,使用pdb调试python程序. 例如,有模拟税收计算的程序:. debug_demo函数计算4500的入账所需的税收. 在需要插入断点的地方,加入红色部分代码:如果_DEBUG值为True,则在该处开始调试(加入_DEBUG的原因是为了方便打开/关闭调试).

ios代码开源

- - CSDN博客移动开发推荐文章
本人从10年开始搞ios开发,从菜鸟到现在的入门,期间遇到了许多困难,也总结了一些东西,本着开源精神,希望大家共同成长的目的把这个工程开源出来.. 这个工程是从11年到13年之前完成的.主要是我平时用到的一些基础功能模块.其中有其他开源的代码和我自己写的一些.代码结构基本乱,12年以后的代码结构还可以,不是很乱,之前水平有限,如果不怎么样就别喷我了.

Oracle错误代码

- - 数据库 - ITeye博客
ORA-00001: 违反唯一约束条件 (.). ORA-00017: 请求会话以设置跟踪事件. ORA-00018: 超出最大会话数. ORA-00019: 超出最大会话许可数. ORA-00020: 超出最大进程数 (). ORA-00021: 会话附属于其它某些进程;无法转换会话. ORA-00022: 无效的会话 ID;访问被拒绝.

Java代码优化

- - ImportNew
2016年3月修改,结合自己的工作和平时学习的体验重新谈一下为什么要进行代码优化. 在修改之前,我的说法是这样的:. 就像鲸鱼吃虾米一样,也许吃一个两个虾米对于鲸鱼来说作用不大,但是吃的虾米多了,鲸鱼自然饱了. 代码优化一样,也许一个两个的优化,对于提升代码的运行效率意义不大,但是只要处处都能注意代码优化,总体来说对于提升代码的运行效率就很有用了.

用 pylint, 写好代码

- Nickcheng - 赖勇浩的编程私伙局
赖勇浩(http://laiyonghao.com). Pylint 是一个 Python 代码分析工具,它分析 Python 代码中的错误,查找不符合代码风格标准(Pylint 默认使用的代码风格是 PEP 8)和有潜在问题的代码. Pylint 是一个 Python 工具,除了平常代码分析工具的作用之外,它提供了更多的功能:如检查一行代码的长度,变量名是否符合命名标准,一个声明过的接口是否被真正实现等等.

完美的代码——Programmers(24)

- 山石 - FeedzShare
来自: 西乔的九卦 - FeedzShare  . 发布时间:2011年06月02日,  已有 2 人推荐. 慢工出细活,只要你要求快,需求分析之类的步骤都只能是过长而已. 载于《程序员》杂志2011年第4期. 这个系列的漫画讲述程序员——这种神秘人类的囧事,故事多来源于我身边的程序员朋友,且以互联网开发背景为主.