单元测试的目的就在于,当你修改了项目中的其中一部分代码A,可能会影响到其他代码B的逻辑,所以我们在写完代码后需要在每一个代码逻辑上都加上单元测试断言,这样就可以提前判断其他代码的逻辑运行是否符合预期,如果不符合预期,就代表你的修改影响了其他代码的逻辑,不能上传,得保证其他代码的逻辑也能正常运行才行。

Junit是xUnit的一个子集,在c++,paython,java语言中测试框架的名字都不相同
xUnit是一套基于测试驱动开发的测试框架
其中的断言机制:将程序预期的结果与程序运行的最终结果进行比对,确保对结果的可预知性
java所用的测试工具是Junit

使用 JUnit 需要导入 JUnit 包。官方网站:https://junit.org/junit5/
在不同编译器下的导包过程不一样,这里以 Maven 为例

  1. <dependency>
  2. <groupId>junit</groupId>
  3. <artifactId>junit</artifactId>
  4. <version>4.12</version>
  5. </dependency>

springboot中使用junit编写单元测试默认是事物回滚的,这样测试的脏数据不影响数据库,即 实际上是执行了对应的插入、删除和更新操作,但是完成操作后执行了事务回滚操作,从而数据库中没有对应数据。

Spring Juint为了不污染数据,对数据的删除和更新操作默认回滚。

在对应的测试方法上添加@Rollback(false)关闭回滚。

  1. 新建一个源代码目录来存放我们的测试代码
  2. 测试类的包应该和被测试类保持一致
  3. 测试方法上必须使用 @Test 进行修饰
  4. 测试方法必须使用 public void 进行修饰,不能带任何的参数
  5. 测试单元中的每个方法必须可以独立测试,测试方法间不能有任何的依赖
  6. 测试类使用 Test 作为类名的后缀(不是必须)
  7. 测试方法使用 test 作为方法名的前缀(不是必须)

目标代码:

  1. package com.imooc.util;
  2. public class Calculate {
  3. public int add(int a,int b) {
  4. return a + b;
  5. }
  6. public int subtract(int a, int b) {
  7. return a - b;
  8. }
  9. public int multiply(int a,int b) {
  10. return a * b;
  11. }
  12. public int divide(int a ,int b) {
  13. return a / b;
  14. }
  15. }

其实就是创建一个JUnit测试文件。

IDEA:
ctrl + shift + t生成测试文件

eclipse:
右键选中需要测试的目标代码,创建 JUnit Test Case 测试用例。




选择一下要测试的方法即可自动生成该方法对应的JUnit空白测试方法。

IDEA中类似操作:代码中右键弹出菜单-Go to-Test
idea2020.2 代码中 Ctrl + Shfit + T

测试代码:

  1. package com.imooc.util;
  2. import static org.junit.Assert.*;
  3. import org.junit.Test;
  4. public class CalculateTest {
  5. /*
  6. * 1.测试方法上必须使用@Test进行修饰
  7. * 2.测试方法必须使用public void 进行修饰,不能带任何的参数
  8. * 3.新建一个源代码目录来存放我们的测试代码
  9. * 4.测试类的包应该和被测试类保持一致
  10. * 5.测试单元中的每个方法必须可以独立测试,测试方法间不能有任何的依赖
  11. * 6.测试类使用Test作为类名的后缀(不是必须)
  12. * 7.测试方法使用test作为方法名的前缀(不是必须)
  13. */
  14. @Test
  15. public void testAdd() {
  16. assertEquals(6, new Calculate().add(3,3));
  17. }
  18. @Test
  19. public void testSubtract() {
  20. assertEquals(3, new Calculate().subtract(5,2));
  21. }
  22. @Test
  23. public void testMultiply() {
  24. assertEquals(4, new Calculate().multiply(2, 2));
  25. }
  26. @Test
  27. public void testDivide() {
  28. assertEquals(3, new Calculate().divide(6, 2));
  29. }
  30. }


注意:

  1. Failure 一般由单元测试使用的断言方法判断失败所引起的,这表示在测试点发现了问题,即 程序输出的结果和我们预期的不一样。
  2. error 是由代码异常引起的,它可以产生于测试代码本身的错误,也可以是被测试代码中的一个隐藏 bug。
  3. 测试用例不说用来证明你是对的,而是用来证明你没有错
  1. package com.bill99.junit;
  2. import java.lang.reflect.Method;
  3. import junit.framework.Assert;
  4. import org.junit.Before;
  5. import org.junit.Test;
  6. public class ACaseTest {
  7. ACase a =null;
  8. @Before
  9. public void setUp() throws Exception {
  10. a = new ACase();
  11. }
  12. @Test
  13. public void testNoParamEchoRequest() throws Exception {
  14. //测试没有参数的echoRequest()方法
  15. Method testNoParamMethod = a.getClass().getDeclaredMethod("echoRequest", null);
  16. //Method对象继承自java.lang.reflect.AccessibleObject,父类方法setAccessible可调
  17. //将此对象的 accessible 标志设置为指示的布尔值。值为 true 则指示反射的对象在使用时应该取消 Java 语言访问检查。值为 false 则指示反射的对象应该实施 Java 语言访问检查。
  18. //要访问私有方法必须将accessible设置为true,否则抛java.lang.IllegalAccessException
  19. testNoParamMethod.setAccessible(true);
  20. //调用
  21. Object result = testNoParamMethod.invoke(a, null);
  22. System.out.println(result);
  23. Assert.assertNotNull(result);
  24. }
  25. @Test
  26. public void testParamEchoRequest() throws Exception {
  27. //测试带有参数的echoRequest(String request)方法
  28. Method testNoParamMethod = a.getClass().getDeclaredMethod("echoRequest",String.class);
  29. testNoParamMethod.setAccessible(true);
  30. //调用
  31. Object result = testNoParamMethod.invoke(a, "this is a test information");
  32. System.out.println(result);
  33. Assert.assertNotNull(result);
  34. }
  35. }

演示代码:

  1. package com.imooc.util;
  2. import static org.junit.Assert.*;
  3. import org.junit.After;
  4. import org.junit.AfterClass;
  5. import org.junit.Before;
  6. import org.junit.BeforeClass;
  7. import org.junit.Test;
  8. public class JunitFlowTest {
  9. /*
  10. * 1.@BeforeClass修饰的方法会在所有方法被调用前被执行,
  11. * 而且该方法是静态的,所以当测试类被加载后接着就会运行它,
  12. * 而且在内存中它只会存在一份实例,它比较适合加载配置文件。
  13. * 2.@AfterClass所修饰的方法通常用来对资源的清理,如关闭数据库的连接
  14. * 3.@Before和@After会在每个测试方法的前后各执行一次。
  15. *
  16. */
  17. @BeforeClass
  18. public static void setUpBeforeClass() throws Exception {
  19. System.out.println("this is beforeClass...");
  20. }
  21. @AfterClass
  22. public static void tearDownAfterClass() throws Exception {
  23. System.out.println("this is afterClass...");
  24. }
  25. @Before
  26. public void setUp() throws Exception {
  27. System.out.println("this is before...");
  28. }
  29. @After
  30. public void tearDown() throws Exception {
  31. System.out.println("this is after");
  32. }
  33. @Test
  34. public void test1() {
  35. System.out.println("this is test1...");
  36. }
  37. @Test
  38. public void test2(){
  39. System.out.println("this is test2...");
  40. }
  41. }

运行结果:

隐藏的固定代码:

  1. @BeforeClass 修饰的方法会在所有方法被调用前被执行,而且该方法是静态的,所以当测试类被加载后接着就会运行它,而且在内存中它只会存在一份实例,它比较适合加载配置文件以及一些只用加载一次的东西。
  2. @AfterClass 所修饰的方法通常用来堆资源的清理,如关闭数据库的连接
  3. @Before@After 会在每个测试方法的前后各执行一次。

    这些都是固定代码,运行时一定会被执行


常用注释:

  • @Test:将一个普通的方法修饰成为一个测试方法

    • @Test(expected=XX.class)
    • @Test(timeout=毫秒)
  • @BeforeClass:它会在所有的方法运行前被执行,static 修饰
  • @AfterClass:它会在所有的方法运行结束后被执行,static 修饰
  • @Before:会在每一个测试方法被运行前执行一次
  • @After:会在每一个测试方法运行后被执行一次
  • @Ignore:所修饰的测试方法会被测试运行器忽略
  • @RunWith:可以更改测试运行器(如想自定义测试运行器可继承 org.junit.runner.Runner)
    可以配合测试套件使用。

测试套件:可以组织测试类一起运行。

  1. 写一个作为测试套件的入口类,这个类里不包含其他的方法
  2. 更改测试运行器为 @RunWith(Suite.class)
  3. 将要测试的类作为数组传入到 @Suite.SuiteClasses({A,B,...})

演示代码:

  1. package com.imooc.util;
  2. import static org.junit.Assert.*;
  3. import org.junit.Test;
  4. import org.junit.runner.RunWith;
  5. import org.junit.runners.Suite;
  6. @RunWith(Suite.class)
  7. @Suite.SuiteClasses({TaskTest1.class,TaskTest2.class,TaskTest3.class})
  8. public class SuiteTest {
  9. /*
  10. * 1.测试套件就是组织测试类一起运行的
  11. *
  12. * 写一个作为测试套件的入口类,这个类里不包含其他的方法
  13. * 更改测试运行器Suite.class
  14. * 将要测试的类作为数组传入到Suite.SuiteClasses({})
  15. */
  16. }

这段代码可以将 TaskTest1、2、3 同时运行测试。从而达到测试套件的效果。

  1. 更改默认的测试运行器为 @RunWith(Parameterized.class)
  2. 声明变量来存放预期值和结果值
  3. 声明一个返回值为 Collection<> 的公共静态方法,并使用 @Parameters 进行修饰
  4. 为测试类声明一个带有参数的公共构造函数,并在其中为之声明变量赋值
  5. 编写测试方法,使用变量

演示代码:

  1. package com.imooc.util;
  2. import static org.junit.Assert.*;
  3. import java.util.Arrays;
  4. import java.util.Collection;
  5. import org.junit.Test;
  6. import org.junit.runner.RunWith;
  7. import org.junit.runners.Parameterized;
  8. import org.junit.runners.Parameterized.Parameters;
  9. @RunWith(Parameterized.class)
  10. public class ParameterTest {
  11. /*
  12. * 1.更改默认的测试运行器为RunWith(Parameterized.class)
  13. * 2.声明变量来存放预期值 和结果值
  14. * 3.声明一个返回值 为Collection的公共静态方法,并使用@Parameters进行修饰
  15. * 4.为测试类声明一个带有参数的公共构造函数,并在其中为之声明变量赋值
  16. */
  17. int expected = 0;//预期值
  18. int input1 = 0;//输入值1
  19. int input2 = 0;//输入值2
  20. public ParameterTest(int expected,int input1,int input2) {
  21. this.expected = expected;
  22. this.input1 = input1;
  23. this.input2 = input2;
  24. }
  25. @Parameters //可以将返回的参数依次放入构造方法中进行测试
  26. public static Collection<Object[]> t() {
  27. return Arrays.asList(new Object[][]{
  28. {3,1,2},//预期值,输入值1,输入值2
  29. {4,2,2}
  30. }) ;
  31. }
  32. @Test
  33. public void testAdd() {
  34. assertEquals(expected, new Calculate().add(input1, input2));
  35. }
  36. }

Spring 配置文件

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans
  3. xmlns="http://www.springframework.org/schema/beans"
  4. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  5. xmlns:p="http://www.springframework.org/schema/p"
  6. xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
  7. <bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
  8. <property name="configLocation" value="classpath:hibernate.cfg.xml"/>
  9. </bean>
  10. <bean id="date" class="java.util.Date"/>
  11. </beans>

Spring 测试代码

  1. package com.imooc.conform;
  2. import java.util.Date;
  3. import org.junit.BeforeClass;
  4. import org.junit.Test;
  5. import org.springframework.context.ApplicationContext;
  6. import org.springframework.context.support.ClassPathXmlApplicationContext;
  7. public class SpringTest {
  8. private static ApplicationContext context = null;
  9. @BeforeClass
  10. public static void setUpBeforeClass() throws Exception {
  11. context = new ClassPathXmlApplicationContext("applicationContext.xml");
  12. }
  13. @Test
  14. public void test() {
  15. Date date = (Date) context.getBean("date");
  16. System.out.println(date);
  17. }
  18. }

第一步 引入 JUnit5 的 jar 包
第二步 创建测试类,使用注解完成

  1. @ExtendWith(SpringExtension.class)
  2. @ContextConfiguration("classpath:bean1.xml")
  3. public class JTest5 {
  4. @Autowired
  5. private UserService userService;
  6. @Test
  7. public void test1() {
  8. userService.accountMoney();
  9. }
  10. }

或者,可以使用一个复合注解替代上面两个注解完成整合

  1. @SpringJUnitConfig(locations = "classpath:bean1.xml")
  2. public class JTest5 {
  3. @Autowired
  4. private UserService userService;
  5. @Test
  6. public void test1() {
  7. userService.accountMoney();
  8. }
  9. }

@Disabled注解:不执行测试

版权声明:本文为blknemo原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/blknemo/p/13346595.html