使用Spring Boot进行单元测试时,发现使用@Autowired注解的类无法自动注入,当使用这个类的实例的时候,报出NullPointerException,即空指针异常。

Spring Boot中的单元测试

先简单说一下Spring Boot中的单元测试。

要在Spring Boot中使用单元测试是很简单的,Spring Boot提供了spring-boot-starter-test的依赖,即JUnit的相关依赖。

  1. 在pom.xml文件中引入依赖:
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-test</artifactId>
</dependency>
  1. 单元测试的语法也很简单,使用@Test注解在方法上或在类上就可以进行简单的单元测试了。
public class aquanTest {
    @Test
    public void testOne () {
        System.out.println("我是一个简单测试。");
    }
}

自动注入serviceidao报空指针异常的原因

说回在JUnit单元测试类中自动注入serviceidao为什么会报空指针异常,报错的原因很明显是 @Autowired 自动注入注解没有成功将类的实例注入。

我们知道,在Spring中,类的实例是交给容器管理的,而在测试类中,容器因为没有相应的上下文环境,是没有办法找到相应的类并进行相应的初始化/实例化操作,因此也就没有在测试类中成功注入类的实例,在试图使用类的实例的情况下也就会报空指针异常的错误。

如何注入serviceidao

根据上面的分析,如果要注入serviceidao的话,我们需要给容器构建一个上下文环境,这里具体是Spring Boot的运行环境。

我们可以通过在测试类上添加两个注解来实现构建上下文环境的效果。

@RunWith(SpringRunner.class)
@SpringBootTest(classes = Application.class)
public class aquanTest {
  
}

我们从注解的命名上就可以看出其功能,但这里还是简单介绍一下。

@RunWith注解

@RunWith注解是类级别的注解, 它提供了一种更改测试运行程序的默认行为的机制。简单理解,@RunWith注解就是一个运行器,其接受一个类的签名来指定是使用什么类、在什么环境下运行。需要注意的是,该注解的参数必须是Runner类的子类,JUnit本身有提供几个Runner,默认值为JUnit4,一个常见的替代方法是参数化类。

当使用@RunWith注解对JUnit测试进行注解时,将对测试的生命周期和测试的运行方式进行一些更改。

比如@RunWith(JUnit4.class)就是指用JUnit4来运行。

比如@RunWith(SpringJUnit4ClassRunner.class)就是让测试运行于Spring测试环境。

比如@RunWith(Suite.class)的话就是一套测试集合。

比如@RunWith(SpringJUnit4ClassRunner.class)使用了Spring的SpringJUnit4ClassRunner,以便在测试开始的时候自动创建Spring的应用上下文。一般情况下想创建spring容器的话,需要通过web.xml配置classloder,但如果注解了@RunWith注解的话,就可以直接使用spring容器,直接使用@Test注解,不用启动spring容器。

在上面的代码中,使用了@RunWith(SpringRunner.class),实际上,SpringRunner类就是继承于Spring的SpringJUnit4ClassRunner类。

@SpringBootTest注解

@SpringBootTest注解是一个Spring Boot提供的注解,通过这个注解可以使JUnit单元测试跑在Spring Boot的运行环境中。

我们可以通过classes参数来指定Spring Boot的启动类。

完整的示例例

最后我们来看一个完整的、在测试类中注入service的示例。

@RunWith(SpringRunner.class)
@SpringBootTest(classes = Application.class)
public class aquanTest {
    @Autowried
    TestService testService;

    @Test
    public void testOne () {
        testService.meow();
    }
}