《编写可读代码的艺术》读书笔记——注释

  注释是代码中很重要的提示,当你在阅读别人的或者自己以前写过的代码时往往能够起到一个“指点迷津”的作用。这篇读书笔记将内容划分为三个部分:

Part 1:哪些情况不需要注释
Part 2:哪些情况需要注释
Part 3:怎么写出言简意赅的好注释

<h3>Part 1:哪些情况不需要注释</h3>

  • 不要为那些从代码本身就能快速推断的事实写注释
    Q:为什么说有的时候没有注释比有注释更好?
    A:这个问题可以用文中的一段开场白来回答。

阅读注释会占用阅读真实代码的时间,并且每条注释都会占用屏幕上的空间。那么,它最好是物有所值的。

下面抛出一段“坏味道”的注释,感受一下:

//The class definition for Account
public Account{
  //Construtor
  Account();

  //set the profit member to a new value
  void setProfit(double profit);

  //return the profit from this Account
  double getProfit();
}

从这段“坏掉”的注释中可以看出来,它所附带的信息对于是毫无价值的,况且代码量又不能用来衡量KPI,所以绝对杜绝写这种注释。

  • 不要为不好名字写注释
    想要写出“可读”的代码,那么为常量、变量、函数起一个好的名字就是第一步(怎么写?以后我将补上,本文只讨论注释)。但是很多时候我们“懒得”去想一个好的名称而不得不补上一堆解释作用的注释。一个好的名字不一个好的注释更重要,注释只会出现多次,而函数或变量却会出现多次。

<h3>Part 2:哪些情况需要注释</h3>
  很多好的注释仅通过“记录你的想法”就能得到,也就是那些你在写代码时的重要想法。从写代码的角度来看,以下几种情况需要注释。

  • ** 加入“导演评论”**

Q:什么是“导演评论”?

“导演评论”就是电影制作者在电影中给出自己的见解并通过讲故事来帮助你理解这部电影是如何制作的。同样,你应该在代码中也加入注释来记录你对代码有价值的见解。

下面给出文中的例子:

//出乎意料的是,对于这些数据用二叉树比用哈希表快40%
//哈希运算的代价比左/右比较大得多

另一个例子:

//作为整体可能会丢掉几个词。这没有问题。要100%解决太难了

当然,注释也可以用来解释为什么代码写得不那么整洁(这可不是说鼓励代码写得很随意),或者是提醒继任者是时候戴上重构的帽子了:

//这个类正在变得越来越乱
//也许我们应该建立一个‘ResourceNode’子类来帮助整理
  • 代码中存在瑕疵
    当代码存在以下问题是,可以采用表中相应的注释:
标记 意义
TODO 还没处理的事
FIXME 已知的无法运行的代码
HACK 对一个问题不得不采用的比较粗糙的解决方案
XXX 危险!这里有重要的问题

给出一个例子:

// TODO 采用更快的算法
  • 常量加注释
    当定义常量时,通常在常量的背后都有一个关于它是什么为什么它是这个值的故事。作为程序员,选择这个某个值的原因往往是读代码的人不知道的,为了避免读者可以去猜测,最好为常量值写上注释。
NUM_THREADS = 8; //设置为1太小了,设置成50又太夸张了

站在读者的角度,以下几种情况也是需要注释的:

  • 意料之中的提问
    对于某些代码,当你意识到别人在读的时候可能产生“为什么这样”的疑问时,就要为这部分代码加上注释了。
//Force vector to relinquish its merory
vector<float>().swap(data);
  • 公布可能的陷阱
    当你意识到代码可能有什么出人意料的情况,也要附上注释。
//运行时间将达到0,所以小心严重嵌套的输入
def FixBrokenHtml(html): …
  • “全局观”注释
    当团队有新成员加入时,具备“全局观”的注释将很快的让能读懂原来的代码。给出一个例子:
//这个文件包含了一些辅助函数,为我们的文件系统提供了更便利的接口
//它处理了文件权限以其他基本的细节
  • “总结性”注释
    “总结性”注释和“全局观”注释有点相似,但它往往是一个类或函数内部的,为多个函数或程序语句做一个总结。给出一个例子:
//数据库连接参数
private String classname="com.mysql.jdbc.Driver";
private String dbConntctedURL = "*******";
private String user = "root";
private String password = "*******";

Part 3:怎么写出言简意赅的好注释###

  • 克服“作者心理阻滞”
    Q:什么是“作者心理阻滞”?
    A:简单来说,就是程序员“懒得”去写的那种“懒”(言简意赅哈哈哈)。
    想要克服“作者心理阻滞”,唯一的办法就是写。刚开始可以先写下心中想的,然后逐渐的提炼,改进。多次循环过后你就达到信手拈来的境界了。下面是书中提到具体操作的几个要点:
  • 声明代码的意图
  • 让注释保持紧凑
  • 采用信息量高的词
  • 润色粗糙的句子
  • 避免使用不明确的代词
    例子:
//Insert the data into the cache, but check if it's too big fist

“it”所指究竟是data还是cache?如果改为下面的就好理解多了

//Insert the data into the cache, but check if the data too big fist
  • 精确的描述函数的行为
    假设你写了一个函数,它统计一个文件中的行数:
//Return the number of lines in this file.
int CountLines(String filename);

上面的注释并不是很精确,因为它存在歧义,或者说还有几种特别的情况:

  • “”(空文件)——0或1行?
  • “hello”——0或1行?
  • “hello\n”——1或2行?
  • “hello\n world”——1或2行?

   但如果你写的注释是这样的:

//Count how many newline bytes('\n') are in the file
int CountLines(String filename);
  • 用输入/输出例子来说明特别的情况
    对于注释,有时候一个精心挑选的输入/输出例子比千言万语还有效。
//Remove the suffix/prefix of 'chars' from the input 'src'
String Strip(String src, String chars)();

看了上面的注释,问题来了:chars是整个要移除的子串,还是一组无序的字母?如果在src结尾有多个chars会怎么样?显然,精心挑选的例子就可以完美的回答问题:

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • //Clojure入门教程: Clojure – Functional Programming for the J...
    葡萄喃喃呓语阅读 3,642评论 0 7
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,605评论 18 399
  • 对象的创建与销毁 Item 1: 使用static工厂方法,而不是构造函数创建对象:仅仅是创建对象的方法,并非Fa...
    孙小磊阅读 1,973评论 0 3
  • 杨霞 和你的相遇仿佛是逐梦人的幻影年少时怀着对另一所学校的憧憬却在辗转中与你相遇艰苦朴素求真务实我大学里的第一课枣...
    爱心驿站阅读 220评论 0 0