代码审查要点简介

在进行团队开发的时候,我们经常要对 pull request 进行代码审查,从而合并不同成员提交的代码。

我们在审查 pull request 的时候可能没有时间去查看代码的整体结构,而着重于被修改的部分。这时,我们可以从以下几个方面入手,检验代码的质量。

对于 Apex 代码的部分,本文参考了官方文库

尽可能的对数据进行批处理

相比较其他编程语言,Apex 最大的特点是在云端执行。当一段代码在云端执行的时候,每次执行都是一个单独的批次。

如果我们在一段代码中需要处理数据,那么我们就要将数据批量的传入代码中,而不是每次处理一条。这样,大量的数据就可以在一个批次中一次处理完,提高执行效率。

简单的说,将函数的参数尽可能的定义为集合类型(列表 List、集合 Set),而不是一个 sObject 类型。

示例:

我们要实现一个将客户名称的后面添加 “Processed” 字样的功能,然后其他的类和功能(比如触发器、API)就可以调用它。

在实现的时候,我们将一个客户对象的列表作为参数传入函数。这样,Apex 代码在云端执行的时候就可以一次处理完多条数据。

public class BulkifyExampleClass {
    public static void BulkifyProcessFunction(List<Account> accounts) {
        for (Account acc : accounts) {
            acc.Name += ' Processed';
        }
    }
}

重复的逻辑

如果一段逻辑出现了两次或更多次,那么我们就必须将这段逻辑单独拎出来,建立一个函数,在不同的地方调用。

这是代码复用最基本的原则:不要重复(Don't Repeat Yourself)。

过度设计

在考虑代码的复用性时,开发者有可能走入另一个极端:过度设计代码的结构。

用户的需求总是会变动,所以我们在设计代码结构的时候需要考虑到未来的扩展性,会想象“如果需求扩展”,该如何更好地维护和修改。

这样考虑的结果有可能导致一些函数过于通用,其中包含了很多逻辑,处理了各种情况。但是在现实生活中,很多情况可能根本不会出现。

过多的“为未来准备”的逻辑会拖慢代码的运行效率。

比如:当一个函数包含了很多可选参数,而这些可选参数会根据不同的情况被使用或不用,那么函数中就会包含复杂的逻辑来判断各种情况。这些逻辑本身就会降低代码的维护性和运行效率。

冗长的函数

业务的流程有时候很复杂,从而导致开发者写出很多的逻辑来实现。如果将这些业务逻辑像流水账一样写在一个函数中,会使这个函数变得冗长。

这时,我们需要将复杂的逻辑拆分成独立的单元,每个单元封装在一个函数中,从而增加代码的可读性。

这种“模块化”思想是软件工程的基本理论之一。在设计项目的整体代码架构时,开发者一般不会犯错误。但是在工作强度很高的时候,开发者赶时间去完成一段复杂的业务逻辑,往往会写出冗长的函数,既难读又难维护。

永远不要相信用户的输入

在代码中,我们会将用户的输入作为参数进行进一步的处理,比如进行数据库的查询。在处理用户输入的时候,我们始终要坚持检查用户的输入,转义它们,永远不要直接使用它们。

SQL 注入、跨站脚本攻击等就是因为相信了用户的输入,从而将恶意的代码引入了原本的代码中,造成破坏。

循环中的逻辑

循环语句是最常用的逻辑语句之一。循环语句会将其内部的逻辑重复执行。

我们要保证在循环语句中的逻辑可以快速执行,而将需要大量时间的逻辑挪到循环外部。

一般来说,代码和数据库通讯需要花费很多时间。当我们有一组数据需要执行相同的逻辑并且存入数据库的时候,我们要保证循环语句中只包含逻辑,而不包含将数据存入数据库的操作。当循环执行完以后,再通过一条语句将这一组数据一次性保存到数据库中。

在 Apex 中,我们可以使用 DML 语句或者 Database 类对数据进行处理。在需要进行这些操作的时候,我们要将它们放在循环语句的外面,从而避免重复的数据库读写操作。

比如:

for (Account acc : accounts) {
    acc.Name += ' Processed';
    update acc; // 多次更新操作,效率低
}

for (Account acc : accounts) {
    acc.Name += ' Processed';
}
update accounts; // 一次更新操作,效率高

避免重复的 SOQL 语句和循环

当我们需要对符合不同条件的同一类型数据进行不同的逻辑处理时,我们可以将它们使用一个 SOQL 语句读取出来,然后分别处理。这种情况多出现在触发器类中。

比如:对于所有新建的客户对象,我们有两个处理逻辑,分别基于它们的名字和雇员数量。

List<Account> bigAccounts = [SELECT Id, Name FROM Account WHERE numberOfEmployees > 500];
for (Account bigAccount : bigAccounts) {
    // 处理客户对象
}

List<Account> specialAccounts = [SELECT Id, Name FROM Account WHERE Name like '%special%'];
for (Account specialAccount : specialAccounts) {
    // 处理客户对象
}

这段代码使用了两次 SOQL 语句,也用了两个循环。我们将其可以优化如下:

List<Account> allAccounts = [SELECT Id, Name, numberOfEmployees FROM Account];
for (Account acc : allAccounts) {
    if (acc.numberOfEmployees > 500) {
        // 处理客户对象
    }
    if (acc.Name.contains('special')) {
        // 处理客户对象
    }
}

这样,将所有逻辑放在一个 SOQL 语句和一个循环中,可以提高代码执行的效率。

不合理的类关联

类和类之间应该保持尽可能少的联系。如果两个类之间的函数相互调用很多,那么可能这两个类本身就需要合并成一个。

尽量少的公用变量和函数

在一个类中,我们需要将公用的变量和函数控制在尽量少的数目。越少的公用部分代表着越少的外界修改权限。

类、函数、变量的命名

类、函数、变量等的命名需要简洁清晰,争取做到让别人不看具体的逻辑就能知道这部分的作用。

参数列表

如果一个函数需要很多参数作为输入,会导致参数列表过长,影响代码的整洁。这时可以定义一个对象,将需要传入的参数整合进对象中,一次传进函数。

比如,有一个函数只包含了创建“地址”的逻辑,它可以被其他代码调用。它拥有很多参数,代表了地址的各个组成部分:

public void createAddress(String streetName, String houseNumber, String postalCode, String city, String province, String country) { 
    //... 
}

在这种情况下,每次调用这个函数,开发者都要传入6个参数,很不方便,也容易出错。

我们可以定义一个“地址”的类 Address,将地址信息的组成部分作为类的成员。然后重写 createAddress 函数,让它只接受一个 Address 类型的参数:

public class Address {
    String streetName;
    String houseNumber;
    String postalCode;
    String city;
    String province;
    String country;

    // ...
}

// ...

public void createAddress(Address a) {
    // ...
}

这样,代码就变得整洁很多。当需要调用 createAddress 函数时,开发者会自己创建合适的 Address 对象,作为参数传入。

注释风格

有一个笑话,说的是“程序员最讨厌两件事:没有注释的别人的代码、为自己的代码写注释”。写注释的最大的误区是没有表达清楚为什么写这段注释,而只是为了注释而注释。

注释是给接手的人看的,不是给机器看的,所以写注释的时候最重要的是讲明为什么写下这段代码,而不是代码本身包含的逻辑。

在为类、函数等组件命名的时候,这些名字大多已经概括了它们的作用和逻辑,其他人接手这段代码时很容易就可以弄清楚。但是很多时候接手的人并不清楚当时写下这段代码的原因,从而不知道这段代码和项目其他部分的联系。在进行修改或者重构的时候牵一发而动全身,造成无法预料的错误。

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

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,602评论 18 399
  • 第2章 基本语法 2.1 概述 基本句法和变量 语句 JavaScript程序的执行单位为行(line),也就是一...
    悟名先生阅读 4,132评论 0 13
  • 〇、前言 本文共108张图,流量党请慎重! 历时1个半月,我把自己学习Python基础知识的框架详细梳理了一遍。 ...
    Raxxie阅读 18,934评论 17 410
  • “我们分手吧。” 对面的晓南明显抖了一下,然后抿着嘴唇重重地喘着气,似乎在压制他的怒气。 我故意忽略了他身后的...
    三分热血少女阿怪阅读 1,419评论 7 21
  • Otto是一个提供事件总线实施的开源项目。组件可以发布和订阅事件。 Otto是Google的Guava事件总线库的...
    ChenXinFei阅读 540评论 0 0