SpringRunner 教程展示了如何使用 SpringRunner 测试 Spring 应用。
Spring 是流行的 Java 应用框架。 在本教程中,我们使用 Spring 5 版本。
SpringRunner
SpringRunner
是SpringJUnit4ClassRunner
的别名,该别名将JUnit
测试库与 Spring TestContext Framework 结合在一起。 我们将其与@RunWith(SpringRunner.class)
一起使用。
使用SpringRunner
,我们可以实现基于 JUnit 4 的标准单元测试和集成测试。
Spring TestContext Framework 提供了通用的,注释驱动的单元和集成测试支持,这些支持与使用中的测试框架(JUnit,TestNG)无关。
SpringRunner 示例
在以下应用中,我们使用SprigRunner
测试一个简单的服务。 该应用是一个 Spring 独立控制台应用。
该应用包含两个属性文件:一个文件用于生产应用,另一个文件用于测试。
pom.xml
src
├───main
│ ├───java
│ │ └───com
│ │ └───zetcode
│ │ │ Application.java
│ │ ├───config
│ │ │ AppConfig.java
│ │ └───service
│ │ HelloService.java
│ └───resources
│ application.properties
│ logback.xml
└───test
├───java
│ └───com
│ └───zetcode
│ └───service
│ HelloServiceTest.java
└───resources
appTest.properties
这是项目结构。
pom.xml
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0
http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.zetcode</groupId>
<artifactId>springrunnerex</artifactId>
<version>1.0-SNAPSHOT</version>
<packaging>jar</packaging>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<maven.compiler.source>11</maven.compiler.source>
<maven.compiler.target>11</maven.compiler.target>
<spring-version>5.1.3.RELEASE</spring-version>
</properties>
<dependencies>
<dependency>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-classic</artifactId>
<version>1.2.3</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>{spring-version}</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-core</artifactId>
<version>{spring-version}</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-test</artifactId>
<version>${spring-version}</version>
</dependency>
<dependency>
<groupId>org.hamcrest</groupId>
<artifactId>hamcrest-all</artifactId>
<version>1.3</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<scope>test</scope>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.codehaus.mojo</groupId>
<artifactId>exec-maven-plugin</artifactId>
<version>1.6.0</version>
<configuration>
<mainClass>com.zetcode.Application</mainClass>
</configuration>
</plugin>
</plugins>
</build>
</project>
这是 Maven 构建文件。 我们具有以下依赖性:logback-classic
用于记录日志,spring-context
和spring-core
是基本的 Spring 依赖性,spring-test
用于测试,hamcrest-all
包含 Hamcrest 匹配库的所有模块,而JUnit
是该库 用于单元测试。
exec-maven-plugin
帮助执行系统和 Java 程序。
resources/logback.xml
<?xml version="1.0" encoding="UTF-8"?>
<configuration>
<logger name="org.springframework" level="ERROR"/>
<logger name="com.zetcode" level="INFO"/>
<appender name="consoleAppender" class="ch.qos.logback.core.ConsoleAppender">
<encoder>
<Pattern>%d{HH:mm:ss.SSS} %blue(%-5level) %magenta(%logger{36}) - %msg %n
</Pattern>
</encoder>
</appender>
<root>
<level value="INFO" />
<appender-ref ref="consoleAppender" />
</root>
</configuration>
logback.xml
是 Logback 日志库的配置文件。
resources/application.properties
app.message=Hello there!
application.properties
包含一个消息属性,由HelloMessage
服务显示。
com/zetcode/AppConfig.java
package com.zetcode.config;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.PropertySource;
@Configuration
@ComponentScan(basePackages = "com.zetcode")
@PropertySource("application.properties")
public class AppConfig {
}
AppConfig
配置组件扫描并从提供的文件中加载属性。
com/zetcode/servide/HelloService.java
package com.zetcode.service;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;
@Service
public class HelloService {
@Value("${app.message}")
private String message;
public String sayHello() {
return message;
}
}
HelloService
返回从application.properties
文件检索到的消息。
com/zetcode/Application.java
package com.zetcode;
import com.zetcode.config.AppConfig;
import com.zetcode.service.HelloService;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.stereotype.Component;
@Component
public class Application {
private static final Logger logger = LoggerFactory.getLogger(Application.class);
public static void main(String[] args) {
var ctx = new AnnotationConfigApplicationContext(AppConfig.class);
var app = ctx.getBean(Application.class);
app.run();
ctx.close();
}
@Autowired
private HelloService helloService;
private void run() {
logger.info("Calling hello service");
logger.info(helloService.sayHello());
}
}
应用使用HelloService
将消息打印到控制台。
$ mvn -q exec:java
17:50:54.118 INFO com.zetcode.Application - Calling hello service
17:50:54.118 INFO com.zetcode.Application - Hello there!
我们运行该应用。
resources/appTest.properties
app.message=Testing hello message
appTest.properties
专用于测试。
com/zetcode/service/HelloServiceTest.java
package com.zetcode.service;
import com.zetcode.config.AppConfig;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.TestPropertySource;
import org.springframework.test.context.junit4.SpringRunner;
import static org.hamcrest.CoreMatchers.equalTo;
import static org.junit.Assert.assertThat;
@RunWith(SpringRunner.class)
@ContextConfiguration(classes={HelloService.class})
@TestPropertySource("/appTest.properties")
public class HelloServiceTest {
@Value("${app.message}")
private String message;
@Autowired
private HelloService helloService;
@Test
public void testHelloMessage() {
var message = helloService.sayHello();
assertThat(message, equalTo(message));
}
}
HelloServiceTest
用于测试HelloService
类。
@RunWith(SpringRunner.class)
@ContextConfiguration(classes={HelloService.class})
@TestPropertySource("/appTest.properties")
public class HelloServiceTest {
测试类用@RunWith(SpringRunner.class)
注释。 @ContextConfiguration
定义了类级别的元数据,用于确定如何加载和配置用于集成测试的应用上下文。 此外,我们还提供了@TestPropertySource
自定义测试属性文件。
@Value("${app.message}")
private String message;
我们从appTest.properties
文件注入消息。
@Autowired
private HelloService helloService;
我们注入HelloMessage
服务类。 这是要测试的类。
@Test
public void testHelloMessage() {
var message = helloService.sayHello();
assertThat(message, equalTo(message));
}
我们测试来自 service 方法的消息是否等于注入的字符串值。
$ mvn -q test
-------------------------------------------------------
T E S T S
-------------------------------------------------------
Running com.zetcode.service.HelloServiceTest
Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0.489 sec
Results :
Tests run: 1, Failures: 0, Errors: 0, Skipped: 0
我们运行测试。
在本教程中,我们展示了如何使用SpringRunner
在 Spring 应用中创建测试。