博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Java程序员必须要知道的单元测试框架Junit详解
阅读量:4035 次
发布时间:2019-05-24

本文共 4475 字,大约阅读时间需要 14 分钟。

作为一名java开发者,相信你或多或少的接触过单元测试,对于测试来讲它是一门能够区分专业开发人员与业余开发人员的重要学科,这篇文章将对java中最常见的一个单元测试框架junit进行一个梳理和讲解。如果你之前没接触过,那么就通过这篇文章进行一个学习。如果你是一个测试老手,我也希望这篇文章能够加深你的印象。

一、为什么需要单元测试

在平时的开发当中,一个项目往往包含了大量的方法,可能有成千上万个。如何去保证这些方法产生的结果是我们想要的呢?当然了,最容易想到的一个方式,就是我们通过System.out来输出我们的结果,看看是不是满足我们的需求,但是项目中这些成千上万个方法,我们总不能在每一个方法中都去输出一遍嘛。这也太枯燥了。这时候用我们的单元测试框架junit就可以很好地解决这个问题。

junit如何解决这个问题的呢?答案在于内部提供了一个断言机制,他能够将我们预期的结果和实际的结果进行比对,判断出是否满足我们的期望。相信到这,你已经迫不及待的想认识一下junit,下面我们直接通过案例,来分析一下这个机制。

二、从案例讲起

1、预备工作

junit4是一个单元测试框架,既然是框架,这也就意味着jdk并没有为我们提供api,因此在这里我们就需要导入相关的依赖。对于IDEA来说,你在构建Maven项目的时候会直接自动添加相关的依赖,如果没有,手动添加即可:

junit
junit
4.12

这里的版本是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方法...afterbefore...test2方法...afterafterClass...

从上面的结果我们来画一张流程图就知道了:

在这里插入图片描述

这个流程相信应该能看懂,如果我们使用过SSM等其他的一些框架,经常会在before中添加打开数据库等预处理的代码,也会在after中添加关闭流等相关代码。

以上这个案例如果能看懂,基本上算是入门了。其实这个案例也比较简单。相信以大家聪明的头脑能看懂。下面我们看看junit中的注解。

三、注解

对于@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
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)); }}

这时候再去测试,只需要去选择相应的值即可,避免了我们一个一个手动输入。

his.expected = expected;

this.input1 = input1;
this.input2 = input2;
}
@Test
public void testAdd() {
assertEquals(expected, new Calculate().add(input1, input2));
}
}
``

这时候再去测试,只需要去选择相应的值即可,避免了我们一个一个手动输入。

对于junit测试,常用的使用方法就是这么多,关于深入了解,只能放在后面的课程中了。今天先到这。

在这里插入图片描述

转载地址:http://kibdi.baihongyu.com/

你可能感兴趣的文章
mint/ubuntu安装搜狗输入法
查看>>
C++动态申请数组和参数传递问题
查看>>
opencv学习——在MFC中读取和显示图像
查看>>
retext出现Could not parse file contents, check if you have the necessary module installed解决方案
查看>>
pyQt不同窗体间的值传递(一)——对话框关闭时返回值给主窗口
查看>>
linux mint下使用外部SMTP(如网易yeah.net)发邮件
查看>>
北京联通华为光猫HG8346R破解改桥接
查看>>
python使用win32*模块模拟人工操作——城通网盘下载器(一)
查看>>
python append 与浅拷贝
查看>>
Matlab与CUDA C的混合编程配置出现的问题及解决方案
查看>>
python自动化工具之pywinauto(零)
查看>>
python一句话之利用文件对话框获取文件路径
查看>>
PaperDownloader——文献命名6起来
查看>>
PaperDownloader 1.5.1——更加人性化的文献下载命名解决方案
查看>>
如何将PaperDownloader下载的文献存放到任意位置
查看>>
C/C++中关于动态生成一维数组和二维数组的学习
查看>>
JVM最简生存指南
查看>>
漂亮的代码,糟糕的行为——解决Java运行时的内存问题
查看>>
Java的对象驻留
查看>>
logback高级特性使用(二) 自定义Pattern模板
查看>>