You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

193 lines
13 KiB
Markdown

This file contains ambiguous Unicode characters!

This file contains ambiguous Unicode characters that may be confused with others in your current locale. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to highlight these characters.

# 26 | 深入浅出之静态测试方法
你好,我是茹炳晟,今天我和你分享的主题是:深入浅出之静态测试方法。
我在分享[《不破不立:掌握代码级测试的基本理念与方法》](https://time.geekbang.org/column/article/14008)这个主题时,系统地介绍了代码级测试常见的五种错误类型(包括语法特征错误、边界行为特征错误、经验特征错误、算法错误,以及部分算法错误),以及对应的四大类测试方法(包括人工静态方法、自动静态方法、人工动态方法,以及自动动态方法)。
今天,我将和你详细讨论人工静态测试方法和自动静态测试方法,来**帮你理解研发流程上是如何保证代码质量的,以及如何搭建自己的自动静态代码扫描方案,并且应用到项目的日常开发工作中去。**
人工静态方法本质上属于流程上的实践,实际能够发现问题的数量很大程度依赖于个人的能力,所以从技术上来讲这部分内容可以讨论的点并不多。但是,这种方法已经在目前的企业级测试项目中被广泛地应用了,所以我们还是需要理解这其中的流程,才能更好地参与到人工静态测试中。
而自动静态方法可以通过自动化的手段以很低的成本发现并报告各种潜在的代码质量问题目前已经被很多企业和项目广泛采用并且已经集成到CI/CD流水线了。作为测试工程师我们需要完成代码静态扫描环境的搭建。接下来我会重点和你分享这一部分内容。
## 人工静态方法
通过我上一次的分析,我们知道了人工静态方法检查代码错误,主要有代码走查、结对编程,以及同行评审这三种手段。那么我们接下来就看一下这三种方法是如何执行的。
* **代码走查Code Review**,是由开发人员检查自己的代码,尽可能多地发现各类潜在错误。但是,由于个人能力的差异,以及开发人员的“思维惯性”,很多错误并不能在这个阶段被及时发现。
* **结对编程Pair Programming**,是一种敏捷软件开发的方法,一般是由两个开发人员结成对子在一台计算机上共同完成开发任务。其中,一个开发人员实现代码,通过被称为“驾驶员”;另一个开发人员审查输入的每一行代码,通常被称为“观察员”。
当“观察员”对代码有任何疑问时,会立即要求“驾驶员”给出解释。解释过程中,“驾驶员”会意识到问题所在,进而修正代码设计和实现。
实际执行过程中,这两个开发人员的角色会定期更换。
* **同行评审Peer Review**是指把代码递交到代码仓库或者合并代码分支Branch到主干Master需要和你同技术级别或者更高技术级别的一个或多个同事对你的代码进行评审只有通过所有评审后你的代码才会被真正递交。
如果你所在的项目使用GitHub管理代码并采用GitFlow的分支管理策略那么在递交代码或者分支合并时需要先递交Pull RequestPR只有这个PR经过了所有评审者的审核才能被合并。这也是同行评审的具体实践。目前只要你采用GitFlow的分支管理策略基本都会采用这个方式。
对于以上三种方式,**使用最普遍的是同行评审**。因为同行评审既能较好地保证代码质量,又不需要过多的人工成本投入,而且递交的代码出现问题后责任明确,另外代码的可追溯性也很好。
结对编程的实际效果虽然不错,但是对人员的利用率比较低,通常被用于一些非常关键和底层算法的代码实现。
## 自动静态方法
自动静态方法,主要有以下三个特点:
* 相比于编译器,可以做到对代码更加严格、个性化的检查;
* 不真正检测代码的逻辑功能,只是站在代码本身的视角,基于规则,尽可能多地去发现代码错误;
* 由于静态分析算法并不实际执行代码,完全是基于代码的词法分析、语法分析、控制流分析等技术,由于分析技术的局限性以及代码写法的多样性,所以会存在一定的误报率。
基于这些特点,自动静态方法通常能够以极低的成本发现以下问题:
* 使用未初始化的变量;
* 变量在使用前未定义;
* 变量声明了但未使用;
* 变量类型不匹配;
* 部分的内存泄漏问题;
* 空指针引用;
* 缓冲区溢出;
* 数组越界;
* 不可达的僵尸代码;
* 过高的代码复杂度;
* 死循环;
* 大量的重复代码块;
*
正是由于自动静态方法具有自动化程度高,检查发现问题的成本低以及能够发现的代码问题广等特点,所以该方法被很多企业和项目广泛应用于前期代码质量控制和代码质量度量。
**在实际工程实践中企业往往会结合自己的编码规范定制规程库并与本地IDE开发环境和持续集成的流水线进行高度整合。**
代码本地开发阶段IDE环境就可以自动对代码实现自动静态检查当代码递交到代码仓库后CI/CD流水线也会自动触发代码静态检查如果检测到潜在错误就会自动邮件通知代码递交者。
接下来,我们一起来看两个自动静态方法发现错误的实际案例,希望可以加深你对自动静态方法的认识。
## 自动静态方法的实际例子
**第一个例子,自动静态方法检查语法特征错误。**
如图1左侧所示的C语言代码存在数组越界的问题一种典型的语法特征错误。
图1右侧就是通过C语言的自动静态扫描工具splint发现的这个问题并给出的分析结果。
![](https://static001.geekbang.org/resource/image/25/85/250302b2a51793aa80663857d6862885.png)
图1 数组越界的错误
**第二个例子,自动静态方法检查内存空间被释放后继续被赋值的错误。**
如图2左侧所示的C语言代码我们用malloc函数申请了一个内存空间并用指针a指向了这个空间然后新建了一个指针b也指向这个空间也就是指针a和指针b实际上指向了同一个内存空间。之后我们把指针a指向的空间释放掉了意味着指针b指向的空间也被释放了。但是此时代码却试图去对指针b指向的空间赋值显然这会导致不可预料的后果。
幸运的是C语言的自动静态扫描工具splint发现了这个问题并给出了详细解释。
![](https://static001.geekbang.org/resource/image/67/19/67979997b21504f5ca9ce99e39bb8d19.png)
图2 内存空间释放后还继续使用的错误
## 实际案例Sonar实战
现在我们已经了解了自动静态代码扫描的基本概念那怎么把这些知识落地到你的实际项目中呢我们就从目前主流的自动静态工具Sonar的使用开始吧。
考虑到你可能以前并没有接触过Sonar所以我会按照step by step的节奏展开。如果你已经用过Sonar了你可以跳过在Mac 电脑上建立Sonar的步骤从完成你的Maven项目的自动静态分析开始。
通过这个Sonar实例你可以掌握
* 搭建自己的SonarQube服务器
* 扫描Maven项目并将结果报告递交到SonarQube服务器
* 在IntelliJ IDE中集成SonarLint插件在IDE中实现实时的自动静态分析
首先,在[Sonar官网](https://www.sonarqube.org/downloads/)下载LTSLong-term Support版本的SonarQube 6.7.5。这里需要注意的是我不推荐在实际工程项目中使用最新版的SonarQube而是建议使用LTS版本以保证稳定性和兼容性。
解压后运行其中的bin/macosx-universal-64目录下的sonar.sh这里需要注意运行sonar.sh时要带上“console”参数。如果执行完成的界面如下图3所示那么说明你的SonarQube服务已经成功启动。
![](https://static001.geekbang.org/resource/image/a4/74/a4b522acf78853ee57aa8e2cbf0c7574.png)
图3 SonarQube启动成功的界面
此时你可以尝试访问localhost:9000并用默认账号用户名和密码都是“admin”登录。
为了简化建立SonarQube的步骤所有的内容我都使用了默认值。比如我直接使用了SonarQube内建的数据库端口也采用了默认的9000。但是在实际工程项目中为了Sonar数据的长期可维护和升级我们通常会使用自己的数据库需要执行下面这些步骤
1. 安装SonarQube之前先安装数据库
2. 建立一个空数据库并赋予CRUD权限
3. 修改SonarQube的conf/sonar.properties中的JDBC配置使其指向我们新建的数据库。我们也可以采用同样的方法来修改默认的端口。
因为要在Maven项目中执行代码静态扫描为此我们需要先找到$MAVEN\_HOME/conf下的settings.xml文件在文件中加入Sonar相关的全局配置具体需要加入的内容如下所示
```
<settings>
<pluginGroups>
<pluginGroup>org.sonarsource.scanner.maven</pluginGroup>
</pluginGroups>
<profiles>
<profile>
<id>sonar</id>
<activation>
<activeByDefault>true</activeByDefault>
</activation>
<properties>
<sonar.host.url>
http://myserver:9000
</sonar.host.url>
</properties>
</profile>
</profiles>
</settings>
```
最后我们就可以在Maven项目中执行“mvn clean verify sonar:sonar”命令完成静态代码扫描。
如果你是第一次使用这个命令那么mvn会自动下载依赖maven-sonar-plugin完成后发起代码的静态扫描并会自动把扫描结果显示到SonarQube中。
图4所示的结果就是我对[《从0到1你的第一个GUI自动化测试》](https://time.geekbang.org/column/article/11913)一文中的GUI测试项目代码的扫描结果。
![](https://static001.geekbang.org/resource/image/67/6f/67e23e2c8a7d44903182d1b1ac38406f.png)
图4 SonarQube的静态扫描结果页面
扫描结果是Passd但同时也发现了三个Code Smell问题或者说是改进建议如图5所示。
1. Class建议放在package中
2. 导入了java.io.BufferedInputStream但没有在实际代码中使用建议删除
3. 建议变量名字不要包含下划线。
![](https://static001.geekbang.org/resource/image/57/89/57e09bf99b6cd9ef1933b54d2e958689.png)
图5 详细扫描结果示例
至此你已经使用Sonar完成了一次代码的静态扫描是不是还挺方便的
但是在日常工作中你可能还想要实时看到Sonar分析的结果这样可以大幅提高修改代码的效率。为此我们可以在IDE中引入SonarLint插件。你可以通过IDE的plugin插件管理界面安装SonarLint。
安装完成后重启IDE你就可以在IDE环境中实时看到Sonar的静态分析结果了如图6所示。
![](https://static001.geekbang.org/resource/image/f5/88/f50c3d4af11f340793afd1cc52078b88.png)
图6 在IDE中直接查看静态扫描结果
另外在IDE中绑定SonarQube就可以把SonarLint和SonarQube集成在一起了如图7所示。集成完成后IDE本地的代码扫描就能使用SonarQube端的静态代码规则库了在企业级的项目中一般要求所有开发人员都使用统一的静态代码规则库所以一般都会要求本地IDE的SonarLint与SonarQube集成。
![](https://static001.geekbang.org/resource/image/70/7f/70d61ce1cc7ec324133d0937309c367f.png)
图7 IDE中的SonarLint和SonarQube绑定
目前自动静态扫描通常都会和持续集成的流水线做绑定最常见的应用场景是当你递交代码后持续集成流水线就会自动触发自动静态扫描这一功能是通过Jenkins以及Jenkins上的SonarQube插件来完成的当你在Jenkins中安装了SonarQube Plugin并且将SonarQube服务器相关的配置信息加入Plugin之后你就可以在Jenkins Job的配置中增加Sonar静态扫描步骤了。
## 总结
人工静态方法,主要有代码走查、结对编程和同行评审三种常用方法。在工程实践中,同行评审因为可以保证代码质量、效率高、责任明确等特点,已经被广泛采用。
自动静态方法,因为自动化程度高、成本低、发现的代码问题广等特点,是常用的代码级测试方法。
在这里测试工程师需要完成代码静态扫描环境的搭建考虑到你以前可能没有接触过Sonar我按照step by step的思路带你一起搭建了一套代码静态扫描环境并分享了一个Maven项目代码静态扫描的实例。
这就是我今天分享的主要内容了,希望可以帮助你解决实际工作中遇到的问题。
## 思考题
除了Sonar你还用过哪些静态代码扫描工具使用过程中遇到过哪些问题
欢迎你给我留言。