JUnit5单元测试框架提供的注解

2024-02-05 03:28

本文主要是介绍JUnit5单元测试框架提供的注解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

  • 第一章、注释在类上的注解
    • 1.1)JUnit5注释在类上的注解
        • 集成测试:@SpringBootTest
        • 集成测试:@ExtendWith(SpringExtension.class)
        • 单元测试:ExtendWith(MockitoExtension.class)
        • 切片测试:@WebMvcTest和@DataJpaTest
        • <font color=red ><b>手动添加bean到测试上下文:@TestConfiguration
    • 1.2)JUnit4类上的注解简单介绍
        • JUnit4集成测试:@RunWith(SpringJUnit4ClassRunner.class)
        • JUnit4单元测试:@RunWith(MockitoJUnitRunner.class)
  • 第二章、注释在成员变量上的注解
    • 2.1)模拟依赖注入
        • @Mock
        • @MockBean
        • @Spy
    • 2.2)注入真实的bean
        • @Autowired
        • @Captor
    • 2.3)模拟注入被测试类的实例:@InjectMocks
        • @InjectMocks
        • 避免@Autowired和@InjectMock同时使用
  • 第三章、注释在方法上的注解
    • 3.1)标记测试
        • @Test:标记测试方法
        • @ParameterizedTest:参数化测试
        • @RepeatedTest:多次重复进行测试
    • 3.2)标记测试顺序
        • @BeforeEach:指定在每个测试方法运行之前执行的方法。
        • @AfterEach:指定在每个测试方法运行之后执行的方法。
        • @BeforeAll:指定在所有测试方法运行之前执行的方法。
        • @AfterAll:指定在所有测试方法运行之后执行的方法。
    • 3.3)其他注解
        • @Disabled:指定禁用测试方法或测试类。
        • @DisplayName:显示测试名称
        • @Tag:指定测试方法的标签。

友情提醒:

先看文章目录,大致了解文章知识点结构,点击文章目录可直接跳转到文章指定位置。有用记得关注

第一章、注释在类上的注解

1.1)JUnit5注释在类上的注解

集成测试:@SpringBootTest

用于加载整个应用程序上下文的注解

@SpringBootTest注解:用于指定Spring Boot应用程序的集成测试。
会加载完整的Spring应用程序上下文,并提供Mock Web Environment,包括所有的bean和配置。
通常用于在JUnit 5中进行Spring Boot应用程序的集成测试。

如果测试时不启动Spring上下文可以进行设置:

//@SpringBootTest(webEnvironment = WebEnvironment.NONE)
//禁用Web环境可以将集成测试转换为更接近于单元测试的测试,而不是测试整个Web应用程序。
@SpringBootTest(webEnvironment = WebEnvironment.NONE)
public class MyIntegrationTest {@Testpublic void testSomething() {// Your test code here}
}
集成测试:@ExtendWith(SpringExtension.class)

@ExtendWith(SpringExtension.class)和@SpringBootTest注解结合使用的效果是在JUnit 5测试中启用Spring的测试框架特性,并且会启动完整的Spring应用程序上下文。

如果你只需要启用Spring支持而不需要加载整个应用程序上下文,可以只单独使用@ExtendWith(SpringExtension.class)。而不使用@SpringBootTest注解,这意味着测试中可以使用Spring的依赖注入、自动装配和其他Spring特性,但无法访问完整的Spring Bean和配置。

单元测试:ExtendWith(MockitoExtension.class)

希望在JUnit 5中使用Mockito框架进行单元测试时可以使用@ExtendWith(MockitoExtension.class)来确保正确的测试环境和行为。以便初始化模拟对象并处理严格的存根。这个扩展类类似于JUnit 4中的MockitoJUnitRunner,它可以在测试类中自动初始化模拟对象,并处理严格的存根。

切片测试:@WebMvcTest和@DataJpaTest

@WebMvcTest用于对Spring MVC控制器进行测试,它会限制测试范围,只会加载与Spring MVC相关的组件,例如控制器、拦截器等,而不会加载整个应用程序上下文。这样可以加快测试速度,并且可以专注于对Web层的测试。


@ExtendWith(SpringExtension.class)
@WebMvcTest(UserController.class)
public class UserControllerTest {@Autowiredprivate MockMvc mockMvc;@Testpublic void testUserController() throws Exception {mockMvc.perform(MockMvcRequestBuilders.get("/users")).andExpect(MockMvcResultMatchers.status().isOk());}
}

@DataJpaTest用于对JPA持久化层进行测试,它会限制测试范围,只会加载与JPA相关的组件,例如实体类、仓储接口等,而不会加载整个应用程序上下文。这样可以加快测试速度,并且可以专注于对持久化层的测试。

@ExtendWith(SpringExtension.class)
@DataJpaTest
public class UserRepositoryTest {@Autowiredprivate TestEntityManager entityManager;@Autowiredprivate UserRepository userRepository;@Testpublic void whenFindByName_thenReturnUser() {// 测试代码}
}
手动添加bean到测试上下文:@TestConfiguration

作用是告诉Spring框架,在测试环境中,这个配置类中定义的bean应该覆盖主应用程序中相同名称的bean,或者提供测试专用的bean定义。

在测试中,定义了一个@TestConfiguration类,通常情况下,它可以放在与被测试类相同的包结构下用于提供测试专用的bean定义。接着通过@Autowired注解将TestConfig配置类引入测试上下文,从而在测试环境中使用其中定义的bean。

    @TestConfigurationstatic class TestConfig {@Beanpublic MyService myService() {return new MyService();}}

通过@Autowired注解将TestConfig配置类引入测试上下文

@SpringBootTest
public class IntegrationTestExample {@Autowiredprivate MyService myService;@Testpublic void testIntegration() {String result = myService.doSomething();assertEquals("expectedResult", result);}}

1.2)JUnit4类上的注解简单介绍

JUnit4集成测试:@RunWith(SpringJUnit4ClassRunner.class)

注意:在JUnit 5中,不再使用@RunWith注解,而是使用更强大的@ExtendWith注解用于扩展测试的行为

@RunWith注解:用于指定JUnit 4测试类的运行器。
可以通过@RunWith注解指定不同的测试运行器,
例如Spring提供的SpringJUnit4ClassRunner
通常用于在JUnit 4中加载Spring上下文进行集成测试。
JUnit4单元测试:@RunWith(MockitoJUnitRunner.class)

注意:告诉JUnit在运行测试时使用Mockito运行器,以便正确处理Mockito的注解和行为,通常用于单元测试中,以便对单个组件进行测试而不涉及外部依赖。

@RunWith(MockitoJUnitRunner.class)
public class YourTestClass {@Testpublic void yourTestMethod() {// Your test method code here}
}

第二章、注释在成员变量上的注解

2.1)模拟依赖注入

@Mock

@Mock:用于模拟依赖项,模拟Spring应用程序上下文中的bean。它会创建一个模拟对象,用于替代应用程序上下文中的真实bean。(mock后真实的方法不再调用)不会执行对象的方法(即使方法报错,test只会使用你设置的返回值,不影响流程)

但用Mockito.when(service.方法名(参数)).thenCallRealMethod();还是可以调真实的方法

@RunWith(MockitoJUnitRunner.class)
public class MyServiceTest {
//模拟注入的依赖项@Mockprivate MyDependency myDependency;@Testpublic void testQueryUserCount(){//待测试的代码。。。
}
}
@MockBean

注解依赖项时,@MockBean注解和@Mock注解的效果是一样的。它们都可以用于模拟依赖项,以便进行单元测试。
@MockBean注解用于模拟Spring应用程序上下文中的bean,并将模拟对象注入到Spring容器中。通常用于模拟service层或repository层的依赖项

@SpringBootTest
public class MyServiceTest {
//MyDependency被用@MockBean注解模拟了,
//可以在不启动Spring Boot应用程序的情况下进行方法的测试。
//也可以使用@Mock注解@MockBeanprivate MyDependency myDependency;@Testpublic void testMyService() {// 测试代码}
}
@Spy

@Spy:用于模拟依赖项,创建一个真实对象的部分模拟。这意味着对象的实际实现将被保留,但您可以选择模拟特定的方法或行为。即会真实的执行对象的方法(如果方法报错,test直接报错)

但用Mockito.doReturn(“不执行此方法”).when(service).方法名(参数);还是可以不调用真实的方法

public class MyServiceTest {@Spyprivate MyDependency myDependency;@Testpublic void testSomething() {// 在这里使用 myDependency 进行测试}
}

2.2)注入真实的bean

@Autowired

用于将实际的Spring bean注入到被测试的类中,以便进行真实的依赖注入。这意味着@Autowired会注入实际的依赖项,而不是模拟对象。

@ExtendWith(SpringExtension.class)
@SpringBootTest
public class MyServiceTest {@Autowiredprivate MyService myService;@Testpublic void testSomething() {// 使用myService进行单元测试}
}
@Captor

@Captor是Mockito框架中的一个注解,用于捕获方法调用时传入的参数,以便在测试中对参数进行断言或验证。
参数捕获:在测试中,使用@Captor可以捕获方法调用时传递的参数,以便在后续的断言中验证参数的值。
参数验证:通过捕获参数,可以对传递给方法的参数进行验证,确保方法得到了期望的参数值。
灵活性:@Captor提供了一种灵活的方式来处理方法调用时的参数,使得测试更加精确和可靠。

// 示例代码中使用@Captor注解捕获参数
@ExtendWith(MockitoExtension.class)
class ExampleTest {@Captorprivate ArgumentCaptor<String> stringCaptor;@Mockprivate SomeClass someClass;@Testvoid testSomething() {// 调用被测试方法someClass.doSomething("test");//verify(someClass)验证someClass对象的doSomething方法是否被调用//@Captor注解创建了一个ArgumentCaptor对象,通过stringCaptor.capture()捕获doSomething方法的参数verify(someClass).doSomething(stringCaptor.capture());//通过stringCaptor.getValue()获取捕获的参数值assertEquals("test", stringCaptor.getValue());}
}

2.3)模拟注入被测试类的实例:@InjectMocks

@InjectMocks

用于创建被测试类的实例,注释的是要测试的实现类并注入模拟对象作为其依赖项。通常用于创建被测试类的实例,并将模拟的依赖项注入到被测试类中,以进行单元测试。

@RunWith(MockitoJUnitRunner.class)
public class UserServiceImplTest {
//需要测试的是UserServiceImpl类,使用 @InjectMocks模拟创建实例@InjectMocksUserServiceImpl userService;
//模拟注入的依赖项@MockUserMapper userMapper;@Testpublic void testQueryUserCount(){//待测试的方法
}
}
避免@Autowired和@InjectMock同时使用

1、在test之前的@BeforeEach注解方法中执行MockitoAnnotations.openMocks(this)
以在每个测试方法之前的初始化操作,包括对Mock对象的初始化。
2、测试的时候要用this.serviceimpl.方法名()测试 否则mock还会使用真实的方法

public class YourTestClass {@InjectMocksprivate YourServiceImpl serviceImpl;@BeforeEachpublic void init() {MockitoAnnotations.openMocks(this);}// 测试方法@Testpublic void testYourServiceMethod() {// 调用被测方法this.serviceImpl.yourServiceMethod();// 断言和验证// ...}
}

第三章、注释在方法上的注解

3.1)标记测试

@Test:标记测试方法

用于标识单元测试方法。JUnit将会执行被@Test注解标记的方法,并对它们进行断言和验证。JUnit 5中的@Test注解使得编写和执行单元测试变得非常简单和直观。

@ParameterizedTest:参数化测试

通过使用 @ParameterizedTest 注解,可以轻松地在单个测试方法中执行多组输入和预期输出的测试。

@ParameterizedTest 注解标记了 testAddition 方法,该方法使用 @CsvSource 提供了多组输入参数。在这个例子中,@CsvSource 提供了三组输入参数,每组参数包括两个加数和预期的结果。在测试方法中,使用提供的输入参数进行计算,并使用断言来验证计算结果是否符合预期。

public class ParameterizedTestExample {@ParameterizedTest@CsvSource({ "1, 1, 2", "2, 3, 5", "5, 5, 10" })void testAddition(int a, int b, int expectedResult) {Calculator calculator = new Calculator();int result = calculator.add(a, b);
//expectedResult期望结果,result实际的计算结果,Lambda表达式,用于生成断言失败时的错误消息。
assertEquals(expectedResult, result, () -> a + " + " + b + " should equal " + expectedResult);}
}
@RepeatedTest:多次重复进行测试

@RepeatedTest 注解用于指定重复测试

public class RepeatedTestExample {
//下面的测试方法将被重复执行五次@RepeatedTest(5)void repeatedTest() {// 测试逻辑assertTrue(true);}
}

3.2)标记测试顺序

@BeforeEach:指定在每个测试方法运行之前执行的方法。

@BeforeEach注解用于标记一个方法,@BeforeEach执行多次,在每个测试方法执行之前都会被执行。这样可以确保在每个测试方法执行前都有一致的初始化操作。

public class ExampleTest {private String message;@BeforeEachpublic void init() {message = "Hello, World!";}@Testpublic void testMessage() {assertEquals("Hello, World!", message);}
}
@AfterEach:指定在每个测试方法运行之后执行的方法。

使用@AfterEach注解的目的是确保在每个测试方法执行之后都有一致的清理操作。这有助于避免测试方法之间的相互影响,以及确保每个测试方法都在一个干净的状态下执行。

public class ExampleTest {private String message;@AfterEachpublic void cleanUp() {message = null;}@Testpublic void testMessage() {message = "Hello, World!";assertEquals("Hello, World!", message);}
}
@BeforeAll:指定在所有测试方法运行之前执行的方法。

@BeforeAll只在第一次运行测试方法时执行一次,后面都不再执行。这意味着它用于执行一次性的全局初始化操作
而@BeforeEach会执行多次,每次运行测试方法都会执行。

public class ExampleTest {private static String message;@BeforeAllpublic static void init() {message = "Hello, World!";}@Testpublic void testMessage() {assertEquals("Hello, World!", message);}
}
@AfterAll:指定在所有测试方法运行之后执行的方法。

@AfterAll注解标记的方法在整个测试类中的所有测试方法都执行之后执行一次,用于执行全局的清理操作。

如果我有4个测试方法,执行了3个,这时候是不会触发@AfterAll注解的方法的。

public class ExampleTest {private static String message;@AfterAllpublic static void cleanUp() {message = null;}@Testpublic void testMessageIsNull() {assertEquals(null, message);}
}

3.3)其他注解

@Disabled:指定禁用测试方法或测试类。

用于禁用单个测试方法或测试类,这意味着被注解的测试方法或测试类将不会被执行。这在临时禁用某些测试时非常有用,例如当测试方法出现问题或需要进行调整时。

public class ExampleTest {@Disabled("This test is currently disabled")@Testvoid disabledTest() {// Test logic that should be disabled}@Testvoid enabledTest() {// Test logic that should be enabled}
}
@DisplayName:显示测试名称

@DisplayName(“Custom Test Name”) 用于指定测试方法的显示名称为 “Custom Test Name”。这个显示名称将会在测试报告、IDE中的测试运行结果以及构建工具(如Gradle、Maven)生成的测试报告中显示。

public class DisplayNameExample {@Test@DisplayName("Custom Test Name")void customTestName() {// 测试逻辑assertEquals(2, 1 + 1);}
}
@Tag:指定测试方法的标签。

在JUnit 5中,使用@Tag注解在测试方法添加标签,也可以为整个测试类添加@Tag注解标签,以便更好地组织和筛选测试。

@Tag("development")
public class ExampleTest {@Test@Tag("unit")void test1() {// 测试方法1的测试逻辑}@Test@Tag("integration")void test2() {// 测试方法2的测试逻辑}
}

通过为测试方法或测试类添加标签,可以更轻松地选择性地运行特定标签的测试,或者排除特定标签的测试。使用Maven在命令行中:

mvn test -Dgroups=tagName
//实例:
mvn test -Dgroups=unit

通过Maven Surefire插件配置选择性运行标签的测试:在maven-surefire-plugin的groups,如果要运行标签为"tagName"的测试,可以将groups参数设置为"tagName"。再执行mvn test命令将会根据配置运行具有指定标签的测试。

<build><plugins><plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-surefire-plugin</artifactId><version>3.0.0-M5</version><configuration><groups>tagName</groups></configuration></plugin></plugins>
</build>

这篇关于JUnit5单元测试框架提供的注解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/679625

相关文章

SpringCloud动态配置注解@RefreshScope与@Component的深度解析

《SpringCloud动态配置注解@RefreshScope与@Component的深度解析》在现代微服务架构中,动态配置管理是一个关键需求,本文将为大家介绍SpringCloud中相关的注解@Re... 目录引言1. @RefreshScope 的作用与原理1.1 什么是 @RefreshScope1.

Spring事务中@Transactional注解不生效的原因分析与解决

《Spring事务中@Transactional注解不生效的原因分析与解决》在Spring框架中,@Transactional注解是管理数据库事务的核心方式,本文将深入分析事务自调用的底层原理,解释为... 目录1. 引言2. 事务自调用问题重现2.1 示例代码2.2 问题现象3. 为什么事务自调用会失效3

Python Dash框架在数据可视化仪表板中的应用与实践记录

《PythonDash框架在数据可视化仪表板中的应用与实践记录》Python的PlotlyDash库提供了一种简便且强大的方式来构建和展示互动式数据仪表板,本篇文章将深入探讨如何使用Dash设计一... 目录python Dash框架在数据可视化仪表板中的应用与实践1. 什么是Plotly Dash?1.1

基于Flask框架添加多个AI模型的API并进行交互

《基于Flask框架添加多个AI模型的API并进行交互》:本文主要介绍如何基于Flask框架开发AI模型API管理系统,允许用户添加、删除不同AI模型的API密钥,感兴趣的可以了解下... 目录1. 概述2. 后端代码说明2.1 依赖库导入2.2 应用初始化2.3 API 存储字典2.4 路由函数2.5 应

Python GUI框架中的PyQt详解

《PythonGUI框架中的PyQt详解》PyQt是Python语言中最强大且广泛应用的GUI框架之一,基于Qt库的Python绑定实现,本文将深入解析PyQt的核心模块,并通过代码示例展示其应用场... 目录一、PyQt核心模块概览二、核心模块详解与示例1. QtCore - 核心基础模块2. QtWid

Java常用注解扩展对比举例详解

《Java常用注解扩展对比举例详解》:本文主要介绍Java常用注解扩展对比的相关资料,提供了丰富的代码示例,并总结了最佳实践建议,帮助开发者更好地理解和应用这些注解,需要的朋友可以参考下... 目录一、@Controller 与 @RestController 对比二、使用 @Data 与 不使用 @Dat

基于@RequestParam注解之Spring MVC参数绑定的利器

《基于@RequestParam注解之SpringMVC参数绑定的利器》:本文主要介绍基于@RequestParam注解之SpringMVC参数绑定的利器,具有很好的参考价值,希望对大家有所帮助... 目录@RequestParam注解:Spring MVC参数绑定的利器什么是@RequestParam?@

最新Spring Security实战教程之Spring Security安全框架指南

《最新SpringSecurity实战教程之SpringSecurity安全框架指南》SpringSecurity是Spring生态系统中的核心组件,提供认证、授权和防护机制,以保护应用免受各种安... 目录前言什么是Spring Security?同类框架对比Spring Security典型应用场景传统

Spring Security注解方式权限控制过程

《SpringSecurity注解方式权限控制过程》:本文主要介绍SpringSecurity注解方式权限控制过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、摘要二、实现步骤2.1 在配置类中添加权限注解的支持2.2 创建Controller类2.3 Us

Java中使用注解校验手机号格式的详细指南

《Java中使用注解校验手机号格式的详细指南》在现代的Web应用开发中,数据校验是一个非常重要的环节,本文将详细介绍如何在Java中使用注解对手机号格式进行校验,感兴趣的小伙伴可以了解下... 目录1. 引言2. 数据校验的重要性3. Java中的数据校验框架4. 使用注解校验手机号格式4.1 @NotBl