`

张砷镓:我所信奉的编程哲学

 
阅读更多

从去年到今年,陆陆续续看完了《代码大全》、《重构》、《代码整洁之道》、《程序员修炼之道》以及《The Art of Readable Code》,获益匪浅。下面就分享几条我赞同并信奉的编程哲学,顺便废话几句。

本文适合有一定编程经验的读者阅读,高手请轻喷:)

 

代码是写给人看的(CODING FOR READING)

 

请先思考,评价一段代码优劣最重要的标准是什么?

有个著名的图,相信大家都见过,讲的是Code Review时被骂WTF的次数越少,代码的质量就越高。这虽然有点无厘头,但是却不无道理。

 

《The Art of Readable Code》中告诉我们,评价一段代码的质量的最佳标准是可读性,即别人理解代码意图所需要的时间。

可读性有什么用,我写出一段代码,能完成目标,能通过测试不就行了吗?

不管你是团队开发,还是一个人单干,只要项目还在运作,代码总是要有人来维护的。如果有一天你不在或者离开了,别人应该能很轻松的看懂你的代码,而不是猜来猜去,最后还要来问你,甚至弃用整段代码重新再写。作为一个负责任的开发者,应该尊重自己的劳动成果,也尊重别人的时间。

当你写下一段代码时,脑子里有清晰的逻辑(比如这里必须这么做,因为……这里不能那么做,因为……)。而别人看这段代码时,脑中是一片空白(这个变量是干什么的?为什么要多加一层判断?这个数为什么是 2 不是 3 ?),除非能完整复现你当时脑中的逻辑,否则就不能理解这段代码。

所以我们应该做的是,把写代码时脑中想到的逻辑的每一个细节,尽可能地写到代码以及注释中,从而帮助后来的阅读者快速复原整个逻辑。简单来说可以有以下的方法:

用常量代替魔术数

在一段代码中出现 if ( level > 3 ) …的时候,也许你会想:为什么是 3 不是 4 呢?这就是所谓的魔术数。通过上下文也许你可以判断出这个 3 是指最高等级,但是可能这段代码中 3 出现了好几次。你会迷惑:它们是不是都是一个意思?当我要把最高等级改成 4 的时候,是不是应该修改所有的 3 ?

如果代码的作者定义一个常量MAX_LEVEL = 3,同时还有一个常量USERS_PER_PAGE = 3,并在不同的地方使用不同的常量,就不会有这样的混淆了。

使用富含信息的类名、变量名和函数名

doit();

$return = $str2 . ':' . $str3;

return $return;

 

这样的代码,简直就是人工混淆过的。你会发现这些代码根本不能帮助你理解代码的含义。也许你该写成这样?

handleError();

$error_message = $error_code . ':' . $error_status;

return $error_message;

 

在注释中说明一段代码存在的原因,而不是行为

$('#item').html(''); //清空item的内容

 

这样的注释有意义么?用自然语言重新描述一次代码的行为,除了徒增维护时的工作量外没有任何价值。你应该说明为什么这样做,以供别人看到这段代码时明白你是怎么想的,并决定如何修改或者对待这段代码。

$('#item').html(''); //先清空容器的内容,否则可能导致内容重复

 

写得越少越好(LESS IS MORE)

 

这个规则的使用性太强了,我简单说说减少逻辑层次和缩小函数体这两个方面吧。

减少逻辑层次

当逻辑层次超过三层时,理解这段代码的难度会急剧上升。我相信谁也不喜欢去读一个n层括号的表达式,或者面对n层缩进的条件判断/循环。

对于复杂的表达式,通过提取中间变量来降低表达式的逻辑层次,保证每个表达式的逻辑层次不超过二层。

对于多层条件判断,大多数情况可以用防御式编程将其简化成单层的条件判断,尽早return或者exit。此外,单行的if-else判断往往可以用三元操作符替换。如果判断实在太多,也许你该重新设计一下结构了。

缩小函数体

记得有一位语言的创始人说过:“我不喜欢比我的头还大的函数”。

事实上,大家都喜欢短小精干、一眼就能看到底的代码。简洁明快的代码有助于别人迅速理解代码的意图,也方便快速定位问题。如果一个函数要滚动屏幕才能看全,那你往往要不断地来回滚动,并强迫自己记住一些信息,再返回去看另一部分,这样做会非常累。

随着函数体不断膨胀,理解它所需要的时间随之增加,出错的几率也会大大提升。而且越大的函数,可维护性和复用性越差。当部分代码逻辑需要修改时,不能快速定位到要修改的位置,也难以确定函数体其他位置是否也需要对应的修改。

当函数尺寸失控时,首先要想到的是,有没有其他方法,用更少的代码完成这个任务?能不能用正则表达式?能不能用查表法?有没有内置的库函数可以利用?

面对一个无法再简化的流程,将其拆分成细粒度的步骤,将每个步骤的相关代码分离出来,提取成子函数,再给子函数起一个漂亮的名字。这样可以降低理解主流程的难度,在做修改时也可以通过函数名快速定位,而且因为相关的代码都在一起,不容易漏改。

也许你会怀疑调用函数所造成的性能损失,我想说现在这个时代,手机都马上四核了……

不要重复(DON'T REPEAT YOURSELF)

 

看过《重构》一书后,我看到代码中任何重复的地方都如见眼中钉。

重复是万恶之源,当你发现你在对代码的不同部分进行同样的修改时就要警惕了。改的地方越多,就越可能出错。也许忘了改一个地方,也许错改或者删除了周边的代码……永远不要让这种事情发生!

将重复的代码提炼成子函数

$('#count').text( + $('#count').text() + 1 );

$('#count').text( + $('#count').text() + 1 );

 

如果一段相同的代码出现两次,基本上你还会第三次用到它,所以很有必要将其提炼成子函数。这样不仅可以减少代码量,还可以降低维护的难度。

countPlusOne();

countPlusOne();

function countPlusOne() {

$('#count').text( + $('#count').text() + 1 ); //以后修改只用改这里就好了

}

 

如果有几段代码很相似,往往可以提取其共性逻辑,使用不同的调用参数进行区分。

countChange( +1 );

countChange( -1 );

function countChange( change ) {

$('#count').text( + $('#count').text() + change );

}

 

用循环减少重复

$item[ 'id' ] = $_POST[ 'id' ];

$item[ 'name' ] = $_POST[ 'name' ];

$item[ 'mail' ] = $_POST[ 'mail' ];

$item[ 'qq' ] = $_POST[ 'qq' ];

 

如果有大块逻辑雷同只有一两个地方更改的代码,往往可以用循环来解决:

foreach ( array( 'id', 'name', 'mail', 'qq' ) as $key ) {

$item[ $key ] = $_POST[ $key ];

}

分享到:
评论

相关推荐

    编程知识库工具 非源码

    该程序仅献于信奉“好记性不如烂笔头”的朋友们! 记性超好!过目不忘者!不需要该程序! 程序作用: 用于大家储存编程和计算机方面的 问题和解决方式!便于大家在查询方便! 程序说明: 大家可根据自己的需要自行...

    高质量C++编程指南

    高质量C++编程指南是一本c++编程圣经,每一个c++程序员手上必备的一本指南性工具。 PS:附上林锐博士的所述前言 软件质量是被大多数程序员挂在嘴上而不是放在心上的东西! 除了完全外行和真正的编程高手外,初读本书...

    高质量C++编程指南.zip

    前 言 6 第1章 文件结构 11 1.1 版权和版本的声明 11 1.2 头文件的结构 12 1.3 定义文件的结构 13 1.4 头文件的作用 13 ...我曾经也这样信奉,并且希望自己成为那样的“真正”的程序员,结果没有得到好下场。

    曾国藩一生信奉16个字.docx

    【曾国藩一生信奉16个字】的文档揭示了曾国藩成功的人生哲学,这些原则对于个人成长和领导力提升具有深远的启示。以下是根据文档内容详细解析的六个核心知识点: 1. **坚忍**:坚忍是面对逆境时保持不屈不挠的精神...

    java8集合源码-usefullinks:有用的编程链接

    有用的编程链接 Java: 精选的 Java 框架、库和软件的精选列表。 Java 对象内部结构 (Shipilev) 在 JVM 厨房中烹饪字节码 Java 性能案例研究 垃圾收集器工作描述(公爵,倒垃圾) 用于在 JVM 上启动和开发 Java 应用...

    Artem711:关于我的信息

    我是谁? ‍:laptop_computer: 全栈TypeScript软件工程师和数学爱好者! :open_book: 语言背景: :Russia: :United_Kingdom: :Spain: :Germany: :books: IB(国际学士学位)学生。 about‍:male_sign_selector:热衷...

    2019_2020学年高中政治第1单元生活智慧与时代精神第1课美好生活的向导第2框关于世界观的学说课堂训练新人教版必修4

    2. **哲学的定义**:哲学是关于世界观的系统化和理论化的学说,它探讨的是整个世界和人生的根本问题。 3. **世界观的种类**:世界观可以是多样的,如“命由天定”或“人定胜天”,这些观念影响个体的行为决策。 4....

    Estruturas-de-Dados---UECE:达斯托斯一世的信奉达人一世

    1. **链表**: 链表是一种线性数据结构,它的元素在内存中不是连续存储的。链表由节点组成,每个节点包含数据和指向下一个节点的引用。Python中虽然没有内置链表类型,但可以通过定义类来实现单链表或双链表。 2. **...

    我的心灵告白 真实的想法加感情

    在人际关系中,我信奉真诚、理解和信任,相信尊重是相互的。我不会强求不属于我的感情,相信一切随缘,顺其自然。面对世俗的眼光,我会用独特的视角去理解,坚持自我,坦然面对。 我愿意在自己的位置上发光发热,...

    计算机专业学生自荐信.pdf

    “顺兮,逆兮,无阻我飞扬”,这是我一直信奉的座右铭。我始终保持着积极向上的态度,对待工作认真负责,并且拥有“如临深渊,如履薄冰”的谨慎作风。诚实守信的个人品质,以及宽广的胸怀和创新的思维,使我面对各种...

    CultistMod:一个在我们当中的mod在游戏中添加了一个新派系-Cultists

    信奉者首先从一个信奉者开始,该信奉者可以将机组人员转变成邪教。 如果邪教组织变得足够大,那么信徒将获胜!mod的工作原理如果您只想安装mod,请转到“ 该国防部在游戏中增加了一个新派系,即“崇拜者”。 信奉者...

    论文研究 - 护士护理方法(NCAp):将差异与阿拉伯世界的伊斯兰信奉患者联系起来

    在护理专业中,护士被多元文化护理能力所吸引,这使护士能够适应并牢记他们如何提供和配合他们的护理工作。 这将使非穆斯林护士在与穆斯林患者的交流方式和交流方式上必须认识并足够敏感。 目的:该研究旨在确定在...

    电子商务大专毕业生自荐信.docx

    我渴望能将我在校所学的理论知识和实践经验应用到实际工作中,同时我也期待在实践中继续学习和成长,成为一名真正有价值的团队成员。 最后,我想表达的是,我对于加入贵公司充满期待和热情。我相信,我的加入能为贵...

    英语GRE单词表

    ##### 1. Abacus - **定义**:算盘,一种古老的计算工具。 - **应用场景**:历史研究、数学教学等领域。 ##### 2. Abandon - **定义**:放弃、遗弃。 - **应用场景**:法律、情感表达等场景。 - **例句**:He ...

    房地产销售技巧售楼员培训资料(睿思).doc

    1. 信奉自己所推销的楼盘:深入了解项目从规划到销售的全过程,这样才能对楼盘充满信心,真诚地帮助客户选择最适合他们的物业。 2. 信任代表的公司:销售人员需对公司有深入的了解和信任,确保公司的诚信和实力,...

    考研政治十年真题解析60pages

    #### 一、马克思主义哲学原理 **1. 马克思主义哲学是科学的世界观和方法论** - **知识点概述**: - **科学实践观**:强调实践是认识的基础,是检验真理的唯一标准。 - **世界观**:对整个世界总的看法和根本观点...

    华为leetcode-gatieme:游戏

    一个信奉“写代码要像写诗一样优美”的90后强迫症程序猿, 奋斗多载, 众里“爬坑”千百夜, 蓦然回首, 才发现, 自己一直将代码写在了水上. 码啥啥不行, 灌水吹逼第一名. 江湖人称“没皮没脸, 天下无敌” :waving_hand:...

    想象龙-信奉-Arduino封面-项目开发

    标题“想象龙-信奉-Arduino封面-项目开发”暗示了一个创新的电子音乐项目,利用Arduino控制器来重现流行歌曲“信奉”(Believer)的封面版本。在这个项目中,开发者可能通过编程Arduino来控制推挽螺线管,这是一种...

    高二语文必修五第2单元古代抒情散文测试题及答案解析精选.doc

    5. 诗人哲学与回归诗学:陶渊明的“归去来兮”代表了一种回归自然、反思人生道路的诗人哲学,与现代人一心向前的发展观形成对比,提醒人们反思现代社会的生存状态。 6. 古代与现代的对照:文章提出,当人类现代性走...

Global site tag (gtag.js) - Google Analytics