多应用+插件架构,代码干净,二开方便,首家独创一键云编译技术,文档视频完善,免费商用码云13.8K 广告
# `SpringRunner`教程 > 原文: [http://zetcode.com/spring/springrunner/](http://zetcode.com/spring/springrunner/) `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 独立控制台应用。 该应用包含两个属性文件:一个文件用于生产应用,另一个文件用于测试。 ```java 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` ```java <?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` ```java <?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` ```java app.message=Hello there! ``` `application.properties`包含一个消息属性,由`HelloMessage`服务显示。 `com/zetcode/AppConfig.java` ```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` ```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` ```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`将消息打印到控制台。 ```java $ 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` ```java app.message=Testing hello message ``` `appTest.properties`专用于测试。 `com/zetcode/service/HelloServiceTest.java` ```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`类。 ```java @RunWith(SpringRunner.class) @ContextConfiguration(classes={HelloService.class}) @TestPropertySource("/appTest.properties") public class HelloServiceTest { ``` 测试类用`@RunWith(SpringRunner.class)`注解。 `@ContextConfiguration`定义了类级别的元数据,用于确定如何加载和配置用于集成测试的应用上下文。 此外,我们还提供了`@TestPropertySource`自定义测试属性文件。 ```java @Value("${app.message}") private String message; ``` 我们从`appTest.properties`文件注入消息。 ```java @Autowired private HelloService helloService; ``` 我们注入`HelloMessage`服务类。 这是要测试的类。 ```java @Test public void testHelloMessage() { var message = helloService.sayHello(); assertThat(message, equalTo(message)); } ``` 我们测试来自`service`方法的消息是否等于注入的字符串值。 ```java $ 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 应用中创建测试。 [Spring MockMvc 教程](/spring/mockmvc/), [Spring `@PropertySource`教程](/spring/propertysource/), [Java 教程](/lang/java/)。