单元测试框架
Java 中,JUnit
和 TestNG
是最受欢迎的单元测试框架。
- JUnit
- TestNG
JUnit
首先是大名鼎鼎的 JUnit ,JUnit 已经成为 Java 应用程序单元测试的事实标准。
JUnit 是一个开源的 Java 语言的单元测试框架,专门针对 Java 设计,使用最广泛。JUnit 目前最新版本是 5
JUnit5 的组成:JUnit 5 = JUnit Platform + JUnit Jupiter + JUnit Vintage
JUnit5 建议使用 Java8 及以上版本
- JUnit Platform 是在 JVM 上启动测试框架的基础,它定义了
TestEngine
在平台运行的新测试框架的 API - JUnit Jupiter 它用于编写测试代码的新的编程和扩展模型。它具有所有新的 Junit 注释和
TestEngine
实现来运行这些注释编写的测试。 - JUnit Vintage JUnit4 已经存在了很长时间,并且有许多以 JUnit4 编写的测试。JUnit Jupiter 还需要支持这些测试。为此,开发了 JUnit Vintage 子项目。提供了一个测试引擎,用于在平台上运行基于 JUnit 3 和 JUnit 4 的测试。它要求 JUnit 4.12 或更高版本出现在类路径或模块路径中。从它的名字 Vintage(古老的;古色古香的)中也能有所体会。
简单例子
我们先来个最简单的例子,别看简单,很多人会犯错
@SpringBootTest
@RunWith(SpringRunner.class)
public class JunitTest {
@Test
public void testJunit(){
System.out.println("junit test");
}
}
复制代码
很简单对吧,如果你用了 SpringBoot 简单到好像没啥说的,其实不然,我们来聊聊:
首先,这段代码使用的是 JUnit 4 还是 JUnit5 ? 你可能会觉得,4 和 5 没啥区别吧,用哪个不一样吗? 代码能跑不就行了?
不是的,4 和 5 肯定有区别这个不用我说了。能跑没问题,但如果你不管是 4 还是 5 都认为一样,API 混用,甚至乱用,那这时候测试出现的各种报错,导致你很懵逼,而且不知道为什么,一通乱查也不知所然。
上面这段代码其实是 JUnit 4 版本,我们看一下 import 就一目了然了,然而可能你在开发的时候没太注意这里是 4 还是 5
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
复制代码
这里确定了,使用的是 4 的版本,这里有几个要注意的点:
@Test
的包是org.junit.Test
,不要搞错了,因为有好几个同名包- 需要
@RunWith(SpringRunner.class)
- 测试类和测试方法需要
public
修饰
我们看下完整的例子:
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
@SpringBootTest
@RunWith(SpringRunner.class)
public class JunitTest {
@Test
public void testJunit(){
System.out.println("junit test ");
}
}
复制代码
这里强调下环境 ,springboot2.2.x 之前支持 JUnit 4
上面有一点提到了 需要 public
修饰的问题,这不很正常吗,为什么要强调?
那是因为 JUnit 5 不需要了,我们看一下用 JUnit 5 来实现的同样的例子 (SpringBoot 2.2.x 之后支持 JUnit 5):
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;
@SpringBootTest
class JunitTest {
@Test
void testJunit5(){
System.out.println("junit5");
}
}
复制代码
这么简单吗? 对,就是这么简单,所以我说 4 和 5 不一样。我们来看区别的地方:
@Test
的包是org.junit.jupiter.api.Test
- 不需要
@RunWith(SpringRunner.class)
- 测试类和测试方法不需要
public
修饰
我见过很多同学在写测试用例时出现的所谓诡异问题,都是因为他自己都没搞清楚用的是 4 还是 5 的情况下将 4 和 5 混用导致的。
如果你的测试用例是 4 ,可以迁移到 5 了,有关 JUnit 4 迁移到 JUnit5
我们再来看一下 pom 依赖这里,你是不是经常看到有关 test 的依赖是这样写的:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
<exclusions>
<exclusion>
<groupId>org.junit.vintage</groupId>
<artifactId>junit-vintage-engine</artifactId>
</exclusion>
</exclusions>
</dependency>
复制代码
为啥? 为什么要排除