在进行团队开发的时候,我们经常要对 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 对象,作为参数传入。
注释风格
有一个笑话,说的是“程序员最讨厌两件事:没有注释的别人的代码、为自己的代码写注释”。写注释的最大的误区是没有表达清楚为什么写这段注释,而只是为了注释而注释。
注释是给接手的人看的,不是给机器看的,所以写注释的时候最重要的是讲明为什么写下这段代码,而不是代码本身包含的逻辑。
在为类、函数等组件命名的时候,这些名字大多已经概括了它们的作用和逻辑,其他人接手这段代码时很容易就可以弄清楚。但是很多时候接手的人并不清楚当时写下这段代码的原因,从而不知道这段代码和项目其他部分的联系。在进行修改或者重构的时候牵一发而动全身,造成无法预料的错误。
网友评论