手把手+零基础带你玩转单元测试,让你的代码更加“强壮”(场景化测试篇)
测试套件
测试套件将多个单元测试用例捆绑在一起并进行集中执行。在JUnit中,可以使用@RunWith和@Suite注解来实现测试套件的运行。
测试案例效果
下面的示例包含TestJunit1和TestJunit2两个测试类,我们将使用测试套件来一起运行这两个测试类:
创建一个类
创建一个待测试的Java类,命名为MessageUtil.java。
/*
* This class prints the given message on console.
*/publicclassMessageUtil{privateString message;publicMessageUtil(String message){this.message = message;}// prints the messagepublicStringprintMessage(){System.out.println(message);return message;}// add "Hi!" to the messagepublicStringsalutationMessage(){
message ="Hi!"+ message;System.out.println(message);return message;}}
创建 Test Case 类
创建一个 java 测试类叫做 TestJunit1.java。
importorg.junit.Test;importorg.junit.Ignore;importstaticorg.junit.Assert.assertEquals;publicclassTestJunit1{String message ="Robert";MessageUtil messageUtil =newMessageUtil(message);@TestpublicvoidtestPrintMessage(){System.out.println("Inside testPrintMessage()");assertEquals(message, messageUtil.printMessage());}}
创建一个 java 测试类叫做 TestJunit2.java。
importorg.junit.Test;importorg.junit.Ignore;importstaticorg.junit.Assert.assertEquals;publicclassTestJunit2{String message ="Robert";MessageUtil messageUtil =newMessageUtil(message);@TestpublicvoidtestSalutationMessage(){System.out.println("Inside testSalutationMessage()");
message ="Hi!"+"Robert";assertEquals(message,messageUtil.salutationMessage());}}
使用 Test Suite 类
- 创建一个名为TestSuite.java的Java类文件,用于执行测试用例。
- 在该类中使用@RunWith(Suite.class)注解。
- 使用@Suite.SuiteClasses注解来引用JUnit测试类。
importorg.junit.runner.RunWith;importorg.junit.runners.Suite;@RunWith(Suite.class)@Suite.SuiteClasses({TestJunit1.class,TestJunit2.class})publicclassJunitTestSuite{}
创建 Test Runner 类
创建一个 java 类文件叫做 TestRunner.java 来执行测试用例。
importorg.junit.runner.JUnitCore;importorg.junit.runner.Result;importorg.junit.runner.notification.Failure;publicclassTestRunner{publicstaticvoidmain(String[] args){Result result =JUnitCore.runClasses(JunitTestSuite.class);for(Failure failure : result.getFailures()){System.out.println(failure.toString());}System.out.println(result.wasSuccessful());}}
验证输出
Inside testPrintMessage()
Robert
Inside testSalutationMessage()
Hi Robert
true
忽略测试
有时候我们的代码还没有准备好,这时运行测试用例可能会导致测试失败。这种情况下,@Ignore注释可以帮助我们。
- 使用@Ignore注释的测试方法将不会被执行。
- 如果一个测试类被@Ignore注释,则该类中的所有测试方法都不会被执行。
现在,我们可以通过一个示例来了解如何使用@Ignore注释。
创建一个类
创建一个将被测试的 java 类命名为 MessageUtil.java。
/*
* This class prints the given message on console.
*/publicclassMessageUtil{privateString message;publicMessageUtil(String message){this.message = message;}// prints the messagepublicStringprintMessage(){System.out.println(message);return message;}// add "Hi!" to the messagepublicStringsalutationMessage(){
message ="Hi!"+ message;System.out.println(message);return message;}}
创建 Test Case 类
创建一个名为 TestJunit 的 Java 测试类,并在其中添加测试方法 testPrintMessage() 和 testSalutationMessage()。在 testPrintMessage() 方法中,添加 @Ignore 注释以忽略该测试方法的执行。
importorg.junit.Test;importorg.junit.Ignore;importstaticorg.junit.Assert.assertEquals;publicclassTestJunit{String message ="Robert";MessageUtil messageUtil =newMessageUtil(message);@Ignore@TestpublicvoidtestPrintMessage(){System.out.println("Inside testPrintMessage()");
message ="Robert";assertEquals(message,messageUtil.printMessage());}@TestpublicvoidtestSalutationMessage(){System.out.println("Inside testSalutationMessage()");
message ="Hi!"+"Robert";assertEquals(message,messageUtil.salutationMessage());}}
创建 Test Runner 类
创建一个 java 类文件叫做 TestRunner.java 来执行测试用例。
importorg.junit.runner.JUnitCore;importorg.junit.runner.Result;importorg.junit.runner.notification.Failure;publicclassTestRunner{publicstaticvoidmain(String[] args){Result result =JUnitCore.runClasses(TestJunit.class);for(Failure failure : result.getFailures()){System.out.println(failure.toString());}System.out.println(result.wasSuccessful());}}
现在运行 Test Runner 类,即不会运行在 Test Case 类中定义的 testPrintMessage() 测试用例。验证输出。testPrintMessage() 测试用例并没有被测试。
InsidetestSalutationMessage()Hi!Roberttrue
TestJunit 在类级别上使用 @Ignore 来忽略所有的测试用例
importorg.junit.Test;importorg.junit.Ignore;importstaticorg.junit.Assert.assertEquals;@IgnorepublicclassTestJunit{String message ="Robert";MessageUtil messageUtil =newMessageUtil(message);@TestpublicvoidtestPrintMessage(){System.out.println("Inside testPrintMessage()");
message ="Robert";assertEquals(message,messageUtil.printMessage());}@TestpublicvoidtestSalutationMessage(){System.out.println("Inside testSalutationMessage()");
message ="Hi!"+"Robert";assertEquals(message,messageUtil.salutationMessage());}}
保持你的 Test Runner 不被改变,如下:
importorg.junit.runner.JUnitCore;importorg.junit.runner.Result;importorg.junit.runner.notification.Failure;publicclassTestRunner{publicstaticvoidmain(String[] args){Result result =JUnitCore.runClasses(TestJunit.class);for(Failure failure : result.getFailures()){System.out.println(failure.toString());}System.out.println(result.wasSuccessful());}}
现在运行 Test Runner 即不会运行在 Test Case 类中定义的任何一个测试样例。
时间测试
Junit 提供了一个便利的暂停选项。如果一个测试用例的执行时间超过了指定的毫秒数,Junit 将自动将其标记为失败。timeout 参数与 @Test 注释一起使用。现在,让我们来了解一下 @Test(timeout) 注释在测试活动中的应用。
创建一个类
- 创建一个在 C:\JUNIT_WORKSPACE 中叫做 MessageUtil.java 的 java 类来测试。
- 在 printMessage() 方法内添加一个无限 while 循环。
创建 Test Case 类
- 创建一个叫做 TestJunit.java 的 java 测试类。
- 给 testPrintMessage() 测试用例添加 1000 的暂停时间。
创建一个文件名为 TestJunit.java 的 java 类。
importorg.junit.Test;importorg.junit.Ignore;importstaticorg.junit.Assert.assertEquals;publicclassTestJunit{String message ="Robert";MessageUtil messageUtil =newMessageUtil(message);@Test(timeout=1000)publicvoidtestPrintMessage(){System.out.println("Inside testPrintMessage()");
messageUtil.printMessage();}@TestpublicvoidtestSalutationMessage(){System.out.println("Inside testSalutationMessage()");
message ="Hi!"+"Robert";assertEquals(message,messageUtil.salutationMessage());}}
创建 Test Runner 类
创建一个文件名为 TestRunner.java 的 java 类来执行测试样例。
importorg.junit.runner.JUnitCore;importorg.junit.runner.Result;importorg.junit.runner.notification.Failure;publicclassTestRunner{publicstaticvoidmain(String[] args){Result result =JUnitCore.runClasses(TestJunit.class);for(Failure failure : result.getFailures()){System.out.println(failure.toString());}System.out.println(result.wasSuccessful());}}
现在运行 Test Runner,它将运行由提供的 Test Case 类中所定义的测试用例。验证输出。testPrintMessage() 测试用例将标记单元测试失败。
Inside testPrintMessage()
Robert
Inside testSalutationMessage()
Hi!Robert
testPrintMessage(TestJunit): test timed out after 1000 milliseconds
false
异常测试
Junit提供了通过代码处理异常的选项,你可以测试代码是否抛出了期望的异常。可以通过在@Test注解中使用expected参数来实现。现在让我们来看一下在@Test注解中使用expected参数的示例。
创建测试用例类
- 创建一个 java 测试类叫做 TestJunit.java。
- 在类中加入一个测试方法 testPrintMessage()。
- 在方法 testPrintMessage() 中加入注释 @Test。
- 在 printMessage()方法中添加一个错误条件。
/*
* This class prints the given message on console.
*/publicclassMessageUtil{privateString message;publicMessageUtil(String message){this.message = message;}// prints the messagepublicvoidprintMessage(){System.out.println(message);int a =0;int b =1/a;}// add "Hi!" to the messagepublicStringsalutationMessage(){
message ="Hi!"+ message;System.out.println(message);return message;}}
创建 Test Case 类
- 创建一个叫做 TestJunit.java 的 java 测试类。
- 给 testPrintMessage() 测试用例添加需要的异常 ArithmeticException。
importorg.junit.Test;importorg.junit.Ignore;importstaticorg.junit.Assert.assertEquals;publicclassTestJunit{String message ="Robert";MessageUtil messageUtil =newMessageUtil(message);@Test(expected =ArithmeticException.class)publicvoidtestPrintMessage(){System.out.println("Inside testPrintMessage()");
messageUtil.printMessage();}@TestpublicvoidtestSalutationMessage(){System.out.println("Inside testSalutationMessage()");
message ="Hi!"+"Robert";assertEquals(message,messageUtil.salutationMessage());}}
验证输出。testPrintMessage() 测试用例将通过。
InsidetestPrintMessage()RobertInsidetestSalutationMessage()Hi!Roberttrue
参数化测试
Junit 4 引入了一个新的功能参数化测试。参数化测试允许开发人员使用不同的值反复运行同一个测试。你将遵循5 个步骤来创建参数化测试。
- 用 @RunWith(Parameterized.class) 来注释 test 类。
- 创建一个由 @Parameters 注释的公共的静态方法,它返回一个对象的集合(数组)来作为测试数据集合。
- 创建一个公共的构造函数,它接受和一行测试数据相等同的东西。
- 为每一列测试数据创建一个实例变量。
- 用实例变量作为测试数据的来源来创建你的测试用例。
一旦每一行数据出现测试用例将被调用。让我们看看活动中的参数化测试。
PrimeNumberChecker
创建一个叫做 PrimeNumberChecker.java 的 java 类来测试。
publicclassPrimeNumberChecker{publicBooleanvalidate(finalInteger primeNumber){for(int i =2; i <(primeNumber /2); i++){if(primeNumber % i ==0){returnfalse;}}returntrue;}}
创建 Parameterized Test Case 类
创建一个叫做 PrimeNumberCheckerTest.java 的 java 类。创建一个文件名为 PrimeNumberCheckerTest.java 的 java 类。
importjava.util.Arrays;importjava.util.Collection;importorg.junit.Test;importorg.junit.Before;importorg.junit.runners.Parameterized;importorg.junit.runners.Parameterized.Parameters;importorg.junit.runner.RunWith;importstaticorg.junit.Assert.assertEquals;@RunWith(Parameterized.class)publicclassPrimeNumberCheckerTest{privateInteger inputNumber;privateBoolean expectedResult;privatePrimeNumberChecker primeNumberChecker;@Beforepublicvoidinitialize(){
primeNumberChecker =newPrimeNumberChecker();}// Each parameter should be placed as an argument here// Every time runner triggers, it will pass the arguments// from parameters we defined in primeNumbers() methodpublicPrimeNumberCheckerTest(Integer inputNumber,Boolean expectedResult){this.inputNumber = inputNumber;this.expectedResult = expectedResult;}@Parameterized.ParameterspublicstaticCollectionprimeNumbers(){returnArrays.asList(newObject[][]{{2,true},{6,false},{19,true},{22,false},{23,true}});}// This test will run 4 times since we have 5 parameters defined@TestpublicvoidtestPrimeNumberChecker(){System.out.println("Parameterized Number is : "+ inputNumber);assertEquals(expectedResult,
primeNumberChecker.validate(inputNumber));}}
现在运行 TestRunner,它将运行由提供的 Test Case 类中所定义的测试用例。
验证输出。
Parameterized Number is : 2
Parameterized Number is : 6
Parameterized Number is : 19
Parameterized Number is : 22
Parameterized Number is : 23
true
版权归原作者 洛神灬殇 所有, 如有侵权,请联系我们删除。