`

如何写出让同事好维护的代码?

阅读更多

注意了,这篇教你如何写出不让同事无法维护的代码!作者:xybaby,原文:https://www.cnblogs.com/xybaby/p/11335829.html

写出整洁的代码,是每个程序员的追求。《clean code》指出,要想写出好的代码,首先得知道什么是肮脏代码、什么是整洁代码;然后通过大量的刻意练习,才能真正写出整洁的代码。

WTF/min是衡量代码质量的唯一标准,Uncle Bob在书中称糟糕的代码为沼泽(wading),这只突出了我们是糟糕代码的受害者。国内有一个更适合的词汇:屎山,虽然不是很文雅但是更加客观,程序员既是受害者也是加害者。

对于什么是整洁的代码,书中给出了大师们的总结:

  • Bjarne Stroustrup:优雅且高效;直截了当;减少依赖;只做好一件事

  • Grady booch:简单直接

  • Dave thomas:可读,可维护,单元测试

  • Ron Jeffries:不要重复、单一职责,表达力(Expressiveness)

其中,我最喜欢的是表达力(Expressiveness)这个描述,这个词似乎道出了好代码的真谛:用简单直接的方式描绘出代码的功能,不多也不少。

命名的艺术

坦白的说,命名是一件困难的事情,要想出一个恰到好处的命名需要一番功夫,尤其我们的母语还不是编程语言所通用的英语。不过这一切都是值得了,好的命名让你的代码更直观,更有表达力。

好的命名应该有下面的特征:

名副其实

好的变量名告诉你:是什么东西,为什么存在,该怎么使用

如果需要通过注释来解释变量,那么就先得不那么名副其实了。

避免误导

  • 不要挂羊头卖狗肉

  • 不要覆盖惯用缩略语

这里不得不吐槽前两天才看到的一份代码,居然使用了 l 作为变量名;而且,user居然是一个list(单复数都没学好!!)

有意义的区分

代码是写给机器执行,也是给人阅读的,所以概念一定要有区分度。

  • 使用读的出来的单词

如果名称读不出来,那么讨论的时候就会像个傻鸟

  • 使用方便搜索的命名

名字长短应与其作用域大小相对应

  • 避免思维映射

比如在代码中写一个temp,那么读者就得每次看到这个单词的时候翻译成其真正的意义

注释

有表达力的代码是无需注释的。

The proper use of comments is to compensate for our failure to express ourself in code.

注释的适当作用在于弥补我们用代码表达意图时遇到的失败,这听起来让人沮丧,但事实确实如此。The truth is in the code, 注释只是二手信息,二者的不同步或者不等价是注释的最大问题。

书中给出了一个非常形象的例子来展示:用代码来阐述,而非注释

bad
// check to see if the employee is eligible for full benefit
if ((employee.flags & HOURLY_FLAG) && (employee.age > 65))

good
if (employee.isEligibleForFullBenefits())

因此,当想要添加注释的时候,可以想想是否可以通过修改命名,或者修改函数(代码)的抽象层级来展示代码的意图。

当然,也不能因噎废食,书中指出了以下一些情况属于好的注释

  1. 法务信息

  2. 对意图的注释,为什么要这么做

  3. 警示

  4. TODO注释

  5. 放大看似不合理之物的重要性

其中个人最赞同的是第2点和第5点,做什么很容易通过命名表达,但为什么要这么做则并不直观,特别涉及到专业知识、算法的时候。

另外,有些第一感觉“不那么优雅”的代码,也许有其特殊愿意,那么这样的代码就应该加上注释,说明为什么要这样,比如为了提升关键路径的性能,可能会牺牲部分代码的可读性。7点建议助您写出优雅的Java代码!这篇也可以看下。

最坏的注释就是过时或者错误的注释,这对于代码的维护者(也许就是几个月后的自己)是巨大的伤害,可惜除了code review,并没有简单易行的方法来保证代码与注释的同步。

函数

函数的单一职责

一个函数应该只做一件事,这件事应该能通过函数名就能清晰的展示。判断方法很简单:看看函数是否还能再拆出一个函数。

函数要么做什么do_sth, 要么查询什么query_sth。最恶心的就是函数名表示只会query_sth, 但事实上却会do_sth, 这使得函数产生了副作用。比如书中的例子

public class UserValidator {
    private Cryptographer cryptographer;
    public boolean checkPassword(String userName, String password) {
        User user = UserGateway.findByName(userName);
        if (user != User.NULL) {
            String codedPhrase = user.getPhraseEncodedByPassword();
            String phrase = cryptographer.decrypt(codedPhrase, password);
            if ("Valid Password".equals(phrase)) {
                Session.initialize();
                return true;
            }
        }
        return false;
    }
}

函数的抽象层级

每个函数一个抽象层次,函数中的语句都要在同一个抽象层级,不同的抽象层级不能放在一起。比如我们想把大象放进冰箱,应该是这个样子的:

def pushElephantIntoRefrige():
    openRefrige()
    pushElephant()
    closeRefrige()

函数里面的三句代码在同一个层级(高度)描述了要完成把大象放进冰箱这件事顺序相关的三个步骤。显然,pushElephant这个步骤又可能包含很多子步骤,但是在pushElephantIntoRefrige这个层级,是无需知道太多细节的。

当我们想通过阅读代码的方式来了解一个新的项目时,一般都是采取广度优先的策略,自上而下的阅读代码,先了解整体结构,然后再深入感兴趣的细节。

如果没有对实现细节进行良好的抽象(并凝练出一个名副其实的函数),那么阅读者就容易迷失在细节的汪洋里。

某种程度看来,这个跟金字塔原理也很像

每一个层级都是为了论证其上一层级的观点,同时也需要下一层级的支持;同一层级之间的多个论点又需要以某种逻辑关系排序。pushElephantIntoRefrige就是中心论点,需要多个子步骤的支持,同时这些子步骤之间也有逻辑先后顺序。

函数参数

函数的参数越多,组合出的输入情况就愈多,需要的测试用例也就越多,也就越容易出问题。

输出参数相比返回值难以理解,这点深有同感,输出参数实在是很不直观。从函数调用者的角度,一眼就能看出返回值,而很难识别输出参数。输出参数通常逼迫调用者去检查函数签名,这个实在不友好。

向函数传入Boolean(书中称之为 Flag Argument)通常不是好主意。尤其是传入True or False后的行为并不是一件事情的两面,而是两件不同的事情时。这很明显违背了函数的单一职责约束,解决办法很简单,那就是用两个函数。

Dont repear yourself

在函数这个层级,是最容易、最直观实现复用的,很多IDE也难帮助我们讲一段代码重构出一个函数。

不过在实践中,也会出现这样一种情况:一段代码在多个方法中都有使用,但是又不完全一样,如果抽象成一个通用函数,那么就需要加参数、加if else区别。这样就有点尴尬,貌似可以重构,但又不是很完美。

造成上述问题的某种情况是因为,这段代码也违背了单一职责原则,做了不只一件事情,这才导致不好复用,解决办法是进行方法的细分,才能更好复用。也可以考虑template method来处理差异的部分。

测试

非常惭愧的是,在我经历的项目中,测试(尤其是单元测试)一直都没有得到足够的重视,也没有试行过TDD。正因为缺失,才更感良好测试的珍贵。

我们常说,好的代码需要有可读性、可维护性、可扩展性,好的代码、架构需要不停的重构、迭代,但自动化测试是保证这一切的基础,没有高覆盖率的、自动化的单元测试、回归测试,谁都不敢去修改代码,只能任其腐烂。

即使针对核心模块写了单元测试,一般也很随意,认为这只是测试代码,配不上生产代码的地位,以为只要能跑通就行了。这就导致测试代码的可读性、可维护性非常差,然后导致测试代码很难跟随生产代码一起更新、演化,最后导致测试代码失效。所以说,脏测试 - 等同于 - 没测试。

因此,测试代码的三要素:可读性,可读性,可读性。

对于测试的原则、准则如下:

  • You are not allowed to write any production code unless it is to make a failing unit test pass. 没有测试之前不要写任何功能代码

  • You are not allowed to write any more of a unit test than is sufficient to fail; and compilation failures are failures. 只编写恰好能够体现一个失败情况的测试代码

  • You are not allowed to write any more production code than is sufficient to pass the one failing unit test. 只编写恰好能通过测试的功能代码

测试的FIRST准则:

  1. 快速(Fast)测试应该够快,尽量自动化。

  2. 独立(Independent) 测试应该应该独立。不要相互依赖

  3. 可重复(Repeatable) 测试应该在任何环境上都能重复通过。

  4. 自我验证(Self-Validating) 测试应该有bool输出。不要通过查看日志这种低效率方式来判断测试是否通过

  5. 及时(Timely) 测试应该及时编写,在其对应的生产代码之前编写

分享到:
评论

相关推荐

    每天写出好代码的5个建议04

    根据给定的文件信息,可以看出文章的主题是关于“每天写出好代码”的建议。然而,提供的部分内容似乎包含了乱码或者是非中文字符,导致无法直接从中提取有意义的信息。因此,本篇文章将依据标题和描述来展开“每天写...

    每天写出好代码的5个建议03

    总之,《每天写出好代码的5个建议03》通过简短的文字传达了提升代码质量的核心理念,即保证代码逻辑的正确性、增强代码的可读性和可维护性,以及持续学习和改进。这些原则适用于所有类型的软件开发项目,无论是个人...

    优美代码力作

    让同事或朋友审查你的代码,可以从不同角度发现潜在问题,促进团队间的知识共享和技能提升。 【代码风格】的统一也很重要,如保持适当间距,统一命名规则,合理组织var语句,适时折叠不常用代码,以及调整编辑器的...

    写程序时编程十大好习惯

    合理的模块划分、函数命名以及注释文档都能让代码更加易于理解和维护。例如: - **模块化**:将复杂的功能分解为独立的模块或组件。 - **函数命名**:采用描述性强的名称来命名函数。 - **注释文档**:为关键部分...

    C语言编程——好习惯.doc

    C语言作为一门基础且强大的编程语言,其编程规范和良好习惯对于写出高效、可维护的代码至关重要。以下是一些在C语言编程中应该遵循的重要原则和习惯: 1. **注释清晰**:尽管有些程序员认为注释是多余的,但良好的...

    编程修养-林锐.pdf

    总之,好的程序员不仅需要了解和掌握编程语言和技术细节,更需要有意识地培养和提升自己的编程修养,从每个函数、每个文件做起,注重每一行代码的编写质量,做到“于细微之处见真功”,才能写出真正高质量的程序作品...

    做程序员需要注意的细节

    作为一名程序员,注重细节是至关重要的。...只有注重每一个环节,才能写出高效、稳定且易于维护的代码,从而成为一名优秀的程序员。通过阅读“重要笔记”中的内容,你将更深入地理解这些细节在实际项目中的应用。

    java重构2.rar part2

    设计模式是解决常见编程问题的模板,如单例模式、工厂模式、观察者模式等,它们能够指导我们写出更加模块化、可扩展的代码,从而降低重构的难度。 总的来说,Java重构是一个持续的过程,需要开发者具备良好的编程...

    XX公司-编程规范与案例

    5. **代码审查**:XX公司可能采用代码审查制度,所有提交的代码都需要经过同事的审查,这有助于发现潜在问题,提升代码质量,同时也促进了团队间的知识分享。 6. **版本控制**:使用版本控制系统如Git,进行代码...

    Becoming a better programmer

    从以上内容中,我们可以总结出成为一个更好的程序员需要掌握的关键知识点: - 编写高质量代码的能力和在日常工作中保持代码质量的意识。 - 有效测试代码的方法,包括单元测试、集成测试等,并在软件开发中融入持续...

    Python夜深人静写算法.docx

    - **同行评审**:邀请同事或朋友对代码进行评审,及时发现潜在问题。 - **重构建议**:基于评审意见对代码进行优化改进。 **八、学习资源** - **在线课程**:Coursera、edX等平台上有很多优质的算法课程。 - **...

    实习大学生个人小结.docx

    同时,保持学习的热情,持续提升专业技能,如设计模式的学习,能帮助我们写出更具可扩展性和维护性的代码,这对于软件开发的长期维护和升级至关重要。 总结,实习不仅是技能的磨炼,更是个人品质和职业素养的塑造。...

    程序设计与数据结构

    良好的程序设计能力能够帮助开发者编写出逻辑清晰、结构合理、易于维护的代码,而合理运用数据结构则是提升程序性能的关键。 首先,程序设计注重的是编写代码的过程。这不仅仅是把代码写出来,而是要考虑到代码的...

    同事整理分享的React.js 面试题大汇总(本人用的).pdf

    - 每种技术(HOC、Render Props、Hooks)都有其适用场景,React的进化旨在提供最佳实践,让开发者更容易写出可维护、可扩展的代码。 以上知识点涵盖了React组件基础、事件处理和组件复用的核心概念,这些都是React...

    Java设计模式 图解 附代码

    通过学习设计模式,开发者不仅能提高代码的可维护性和可扩展性,还能提升自己的编程技巧和设计思路。 #### 二、创建型模式 创建型模式关注的是对象的创建机制,它们提供了一种创建对象的方式,以便系统可以独立于...

    编程修养(程序员需要注意的关于编程的注意事项)

    【编程修养】是每个程序员在职业生涯中必须重视和不断提升的一项能力。编程修养并不仅仅是掌握技术细节或者编程速度,更重要的是...通过不断提高编程修养,程序员不仅能写出优秀的代码,还能成为团队中的宝贵财富。

    热乎乎的完整版react面试题贼细+答案(同事面试用的).pdf

    React是JavaScript库,用于构建用户界面,特别是单页应用程序(SPA)。面试中,React的知识点通常涵盖组件化、事件处理、状态管理等多个方面。...在实际开发中,熟练运用这些机制可以写出高效、可扩展的React应用。

    c语言编程修养之道(一位工程师的经验之谈)

    在《C语言编程修养之道(一位工程师的经验之谈)》这篇文章中,作者分享了自己在程序开发过程中积累的宝贵经验,总结出了三十二条提高编程水平的原则,旨在帮助读者写出高质量、易读且易于维护的代码。 #### 1. ...

Global site tag (gtag.js) - Google Analytics