代码之美——像写作一样去coding

作为程序员,我们或许常常会被问到:你都学过什么语言呢?你最擅长的是哪一门语言?是的,一门语言。

这里所提到的语言并非我们的母语汉语,也不是英语亦或其他任何一种用于交流日常工作生活的语言。而是指编程过程中,连通人与机器、人与人之间的一种表达方式。让机器读懂代码很简单,只需注明所用代码的语言规则就好,毕竟机器那么聪明 :)但是如果想要让其他人看懂,就不能这样简单粗暴了。人是感性与理性结合的动物,优雅“风趣”的表达能够让对方更快、更轻松的读懂你的代码。

既然都是表达内容,那么为什么不用写文章的方式来写代码呢?文章是人们日常用于交流表达的一种方式,那么我们是否可以吸收文章的优势来用在写代码上呢?

image

让句子连在一起组成段落

我们可以试着把方法抽象成文章里的一句话,方法内紧接着调用的另一个方法,就好像是第一句话还需要第二句话去完善一样。所以我们应该把句子2放在句子 1 后面,也就是说我们可以把被调用的方法放在调用方法下面。

同理,一个方法内部两个相邻方法的调用先后顺序就像是文章里两个相邻句子的先后顺序一样。所以我们也应把这种顺序作为方法上下排列的顺序。

那么如果我们不遵循这种规则会怎么样呢?

private void preparePizza(Pizza pizza) {
  getFlour();
}

private void boxPizza(Pizza pizza) {
...
}

public Pizza orderPizza(String type) {
  Pizza pizza = getBasePizza(type);
  preparePizza(pizza);
  boxPizza(pizza);
  return pizza;
}

private Flour getFlour() {
...
}

private Pizza getBasePizza(String type) {
...
} 

上面这段代码方法排序是随意的,我们无法直观的看到方法的执行顺序。就好像是:“再然后我去吃早饭;然后我去洗漱;我早上七点起床”,混乱的顺序增加了我们理解代码的困难度。

如果我们遵循这两种规则来排序方法,那就如下面这样:

public Pizza orderPizza(String type) {
  Pizza pizza;
  pizza = getPizza(type);

  preparePizza();
  boxPizza();
  return pizza;
}

private Pizza getPizza(String type) {
...
}

private void preparePizza() {
  getFlour();
}

private Flour getFlour() {
...
}

private void boxPizza() {
...
}

当我们阅读这段代码时,会觉得这是一个整体,只需要向读文章一样,上下滑动阅读即可。

有的人可能会说,通过快捷键一样可以定位到下一个方法。但是快捷键仅适用于逻辑简单的情况,在复杂的逻辑中来回定位所产生的上下跳跃会让人觉得非常难受,这也是我们应当竭力避免的。

image

定义小范围章节目录

一本书或一篇长文,一般都会有章节目录。就好像一个类中有几个提供给外界调用的public方法,这可以使我们有很好的全局观。所以我们应该把类中一些提供主要功能的对外方法放到一起,这些方法要以功能相近来集聚。

如下:

@Override
public ResultT getAResult(KeyT keySearch) {
  ...
}

@Override
public ResultT getAResultUntilTimeout(KeyT keyT, long timeout, TimeUnit timeUnit) throws TimeoutException {
  ...
}

@Override
public List<ResultT> getResultsUntilOneTimeout(KeyT keyT, long timeout, TimeUnit unit) {
  ...
}

@Override
public List<ResultT> getResultsUntilTimeout(KeyT keyT, long timeout, TimeUnit unit) {
  ...
}

@Override
public List<ResultT> getResultsUntilEnoughOrTimeout(KeyT keyT, int expectNum, long timeout, TimeUnit unit) {
  ...
}

@Override
public List<ResultT> getResultsUntilEnoughOrOneTimeout(KeyT keyT, int expectNum, long timeout, TimeUnit unit) {
  ...
}

@Override
public List<ResultT> getResultsUntilEnough(KeyT keyT, int expectNum) throws TimeoutException {
  ...
}

这是我写的一个search-framework中的部分代码,这些方法都是相近的,所以把它们放到一起。另外我们要小范围的集聚,即把相似的开放式方法放在一起,这些方法的下面就是内部调用的方法,继续遵循“让句子连在一起组成段落”的理念。

image.gif

其实有一种更好的办法,就是可以用一种插件让IDEA自动生成一种目录式方法。这种方法只包含基本信息,没有内部实现,并且我们可以点击目录进入方法的准确位置(准确位置的方法排序遵循段落式描述法)。至于如何让IDEA知道哪些方法应该生成目录式方法,我们或许可以通过某种注解去定义。

那么它看起来就好像下面这样:

public class ConcurrentEntirelySearch<KeyT, ResultT, PathT> implements EntirelySearch<KeyT, ResultT> {
  private static final long MAX_WAIT_MILLISECOND = 1000 * 60 * 2;

  private final List<PathT> rootCanBeSearch;
  private final ConcurrentEntirelyOpenSearch<KeyT, ResultT, PathT> openSearch;

  public ConcurrentEntirelySearch(List<PathT> rootCanBeSearch, SearchModel searchModel) {
    this.rootCanBeSearch = rootCanBeSearch;
    this.openSearch = new ConcurrentEntirelyOpenSearch<>(searchModel);
  }

/** 目录(如何展示细节待设计)*/
  @Override
--- public ResultT getAResult(KeyT keySearch) {...}

  @Override
--- public ResultT getAResultUntilTimeout(KeyT keyT, long timeout, TimeUnit timeUnit) throws TimeoutException {...}

  @Override
--- public List<ResultT> getResultsUntilOneTimeout(KeyT keyT, long timeout, TimeUnit unit) {...}

  @Override
--- public List<ResultT> getResultsUntilTimeout(KeyT keyT, long timeout, TimeUnit unit) {...}
/** 目录完(虚拟内容,可点击跳转至方法)------------------- */

  @Override
  public ResultT getAResult(KeyT keySearch) { // 此为真实方法,非目录
    methodA();      //方法排序遵循上述的 段落式描述法
    methodB();
  }

  private void methodA() {
  ...
  }

  private void methodB() {
  ...
  }
  //下同,方法内调用的方法略
  @Override
  public ResultT getAResultUntilTimeout(KeyT keyT, long timeout, TimeUnit timeUnit) throws TimeoutException {
    ...
  }

  @Override
  public List<ResultT> getResultsUntilOneTimeout(KeyT keyT, long timeout, TimeUnit unit) {
    ...
  }   @Override
  public List<ResultT> getResultsUntilTimeout(KeyT keyT, long timeout, TimeUnit unit) {
    ...
  }
}

这些目录我认为应该放在构造方法的下面,这样看起来更加有条理。


写文章时不要让每一行过长

相信没有人愿意去看由一行行长文所组成的段落,长度适中的段落能够让读者在跳行时有一个休息,也给大脑一个轻微的缓冲,这样的阅读舒适感会高很多。所以我们要善于利用这个度,不要让代码过长,但是有时候也可以利用这个度去做inline,只要不超过那个限度就ok。

image

这个思想是在一次ThoughWorks的活动中受到的启发,inline是很好,但是它不能过度,只要我们遵循“写文章时不要让每一行过长”的理念就ok。让读者得以take a breath。

就好像下面这次重构一样:

//重构前
@Test
public void should_return_1B_given_1000000_length() {
  Gold gold = new Gold(1000000);
  String length = gold.getLength();
  assertEquals("1B", length);
}
//重构后
@Test
public void should_return_1B_given_1000000_length() {
  assertEquals("1B", new Gold(1000000).getLength());
}

上面这个例子就利用了这种理念,在读者读一行代码时,能接受的最多字符是有限的,过长就会产生疲倦感、厌恶感。

下面来看一个反例:

//重构前
int previousNumber = getNumberByUnit(lastIndex);
String target = numberString.substring(0, lastIndex);
compute(Long.parseLong(target), previousNumber);
//重构后
int previousNumber = getNumberByUnit(lastIndex);
compute(Long.parseLong(numberString.substring(0, lastIndex)), previousNumber);

这里有必要解释一下“度”的概念,我认为度不应该以每一行能容纳的字符数来衡量。而是要以 该行内变量或方法命名的长度、该行内嵌套调用的方法数量、该行内调用方法的参数数量 这三点综合去考虑。

“某一行命名比较长” 、 “某一行嵌套调用的方法比较多” 和 “某一行方法的参数比较多” 所能承受的最大字符数是不一样的。比如:读者能接受的“命名比较长”的最大长度跟所能接受的 “调用方法多的” 最大长度所能容纳的字符数肯定不一样,因为命名就算再长点也还像是一句话,我们也还算可以理解,而调用的方法逐渐变多那理解的复杂度就会几何增长。


总结

如文载道,要想让自己的代码发挥更大的影响,就一定要花时间去琢磨怎么把它写的更易读。我们应坚持写“笨”代码的思想,如果代码能像文章那样有条理,有规律可循,那无疑可以增强代码的可维护性。这样的代码阅读起来也会让人更加舒适。

文/ThoughtWorks 郑占鹏

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

推荐阅读更多精彩内容