作为一名测试工程师,相信你或多或少的接触过单元测试,对于测试来讲它是一门能够区分专业开发人员与业余开发人员的重要学科,这篇文章将对java中最常见的一个单元测试框架junit进行一个梳理和讲解。如果你之前没接触过,那么就通过这篇文章进行一个学习。如果你是一个测试老手,我也希望这篇文章能够加深你的印象。
一、为什么需要单元测试
在平时的开发当中,一个项目往往包含了大量的方法,可能有成千上万个。如何去保证这些方法产生的结果是我们想要的呢?当然了,最容易想到的一个方式,就是我们通过System.out来输出我们的结果,看看是不是满足我们的需求,但是项目中这些成千上万个方法,我们总不能在每一个方法中都去输出一遍嘛。这也太枯燥了。这时候用我们的单元测试框架junit就可以很好地解决这个问题。
junit如何解决这个问题的呢?答案在于内部提供了一个断言机制,他能够将我们预期的结果和实际的结果进行比对,判断出是否满足我们的期望。相信到这,你已经迫不及待的想认识一下junit,下面我们直接通过案例,来分析一下这个机制。
二、从案例讲起
1、预备工作
junit4是一个单元测试框架,既然是框架,这也就意味着jdk并没有为我们提供api,因此在这里我们就需要导入相关的依赖。对于IDEA来说,你在构建Maven项目的时候会直接自动添加相关的依赖,如果没有,手动添加即可:
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
</dependency>
这里的版本是4.12。当然还有最新的版本。你可以手动选择。这里选用的是4的版本。
2、案例
这里我们要测试的功能超级简单,就是加减乘除法的验证。
public class Calculate {
public int add(int a,int b) {
return a + b;
}
public int subtract(int a, int b) {
return a - b;
}
public int multiply(int a,int b) {
return a * b;
}
public int divide(int a ,int b) {
return a / b;
}
}
然后我们看看如何使用junit去测试。
public class CalculateTest {
@Test
public void testAdd() {
assertEquals(2, new Calculate().add(1,1));
}
@Test
public void testSubtract() {
assertEquals(8, new Calculate().subtract(10,2));
}
@Test
public void testMultiply() {
assertEquals(6, new Calculate().multiply(3, 2));
}
@Test
public void testDivide() {
assertEquals(5, new Calculate().divide(10, 2));
}
}
以上就是我们的单元测试,需要遵循一下规则:
1、每一个测试方法上使用@Test进行修饰
2、每一个测试方法必须使用public void 进行修饰
3、每一个测试方法不能携带参数
4、测试代码和源代码在两个不同的项目路径下
5、测试类的包应该和被测试类保持一致
6、测试单元中的每个方法必须可以独立测试
以上的6条规则,是在使用单元测试的必须项,当然junit也建议我们在每一个测试方法名加上test前缀,表明这是一个测试方法。
assertEquals是一个断言的规则,里面有两个参数,第一个参数表明我们预期的值,第二个参数表示实际运行的值。不过junit5对这些做出了一些改变,我们会在后续的文章中专门介绍。
我们运行一下测试类,就会运行每一个测试方法,我们也可以运行某一个,只需要在相应的测试方法上面右键运行即可。如果运行成功编辑器的控制台不会出现错误信息,如果有就会出现failure等信息。
3、运行流程
在上面的每一个测试方法中,代码是相当简单的,就一句话。现在我们分析一下这个测试的流程是什么:
public class JunitFlowTest {
@BeforeClass
public static void setUpBeforeClass() throws Exception {
System.out.println("beforeClass...");
}
@AfterClass
public static void tearDownAfterClass() throws Exception {
System.out.println("afterClass...");
}
@Before
public void setUp() throws Exception {
System.out.println("before...");
}
@After
public void tearDown() throws Exception {
System.out.println("after");
}
@Test
public void test1() {
System.out.println("test1方法...");
}
@Test
public void test2(){
System.out.println("test2方法...");
}
}
在上面的代码中,我们使用了两个测试方法,还有junit运行整个流程方法。我们可以运行一下,就会出现下面的运行结果:
beforeClass...
before...
test1方法...
after
before...
test2方法...
after
afterClass...
从上面的结果我们来画一张流程图就知道了:
这个流程相信应该能看懂,如果我们使用过SSM等其他的一些框架,经常会在before中添加打开数据库等预处理的代码,也会在after中添加关闭流等相关代码。
以上这个案例如果能看懂,基本上算是入门了。其实这个案例也比较简单。相信以大家聪明的头脑能看懂。下面我们看看junit中的注解。
现在我也找了很多测试的朋友,做了一个分享技术的交流群,共享了很多我们收集的技术文档和视频教程。
如果你不想再体验自学时找不到资源,没人解答问题,坚持几天便放弃的感受
可以加入我们一起交流。而且还有很多在自动化,性能,安全,测试开发等等方面有一定建树的技术大牛
分享他们的经验,还会分享很多直播讲座和技术沙龙
可以免费学习!划重点!开源的!!!
qq群号:310357728【暗号:csdn999】
三、注解
对于@Test,里面有很多参数供我们去选择。我们来认识一下
1、@Test(expected=XX.class)
这个参数表示我们期望会出现什么异常,比如说在除法中,我们1/0会出现ArithmeticException异常,那这里@Test(expected=ArithmeticException.class)。在测试这个除法时候依然能够通过。
2、@Test(timeout=毫秒 )
这个参数表示如果测试方法在指定的timeout内没有完成,就会强制停止。
3、@Ignore
这个注解其实基本上不用,他的意思是所修饰的测试方法会被测试运行器忽略。
4、@RunWith
更改测试运行器。
四、测试套件
在文中一开始我们曾经提到,如果我们的项目中如果有成千上万个方法,那此时也要有成千上万个测试方法嘛?如果这样junit使用起来还不如System.out呢,现在我们认识一下测试嵌套的方法,他的作用是我们把测试类封装起来,也就是把测试类嵌套起来,只需要运行测试套件,就能运行所有的测试类了。
//这里有很多个测试类
public class Test1 {
@Test
public void test() {
System.out.println("测试类1");
}
}
public class Test2 {
@Test
public void test() {
System.out.println("测试类2");
}
}
//这里一次可以类推
下面我们使用测试套件,把这些测试类嵌套在一起。
@RunWith(Suite.class)
@Suite.SuiteClasses({Test1.class,Test2.class等相关测试类})
public class SuiteTest {
/*
* 写一个空类:不包含任何方法
* 更改测试运行器Suite.class
* 将测试类作为数组传入到Suite.SuiteClasses({})中
*/
}
也很简单,下面我们看一下,参数化设置。
五、参数化设置
什么是参数化设置呢?在一开始的代码中我们看到,测试加法的时候是1+1,不过我们如果要测试多组数据怎么办?总不能一个一个输入,然后运行测试吧。这时候我们可以把我们需要测试的数据先配置好。
@RunWith(Parameterized.class)
public class ParameterTest {
int expected =0;
int input1 = 0;
int input2 = 0;
@Parameters
public static Collection<Object[]> t() {
return Arrays.asList(new Object[][]{
{3,1,2},
{4,2,2}
}) ;
}
public ParameterTest(int expected,int input1,int input2) {
this.expected = expected;
this.input1 = input1;
this.input2 = input2;
}
@Test
public void testAdd() {
assertEquals(expected, new Calculate().add(input1, input2));
}
}
这时候再去测试,只需要去选择相应的值即可,避免了我们一个一个手动输入。
对于junit测试,常用的使用方法就是这么多,关于深入了解,只能放在后面的课程中了。今天先到这。
END今天的分享就到此结束了,点赞关注不迷路~