为什么组件扫描对Spring Boot单元测试不起作用?

服务类FooServiceImpl使用@Service aka

注释,@Component这使其可以自动装配。为什么在单元测试期间不选择此类并进行自动布线?

@Service

public class FooServiceImpl implements FooService {

@Override

public String reverse(String bar) {

return new StringBuilder(bar).reverse().toString();

}

}

@RunWith(SpringRunner.class)

//@SpringBootTest

public class FooServiceTest {

@Autowired

private FooService fooService;

@Test

public void reverseStringShouldReverseAnyString() {

String reverse = fooService.reverse("hello");

assertThat(reverse).isEqualTo("olleh");

}

}

测试未能加载应用程序上下文,

2018-02-08T10:58:42,385 INFO    Neither @ContextConfiguration nor @ContextHierarchy found for test class [io.github.thenilesh.service.impl.FooServiceTest], using DelegatingSmartContextLoader

2018-02-08T10:58:42,393 INFO Could not detect default resource locations for test class [io.github.thenilesh.service.impl.FooServiceTest]: no resource found for suffixes {-context.xml}.

2018-02-08T10:58:42,394 INFO Could not detect default configuration classes for test class [io.github.thenilesh.service.impl.FooServiceTest]: FooServiceTest does not declare any static, non-private, non-final, nested classes annotated with @Configuration.

2018-02-08T10:58:42,432 INFO Loaded default TestExecutionListener class names from location [META-INF/spring.factories]: [org.springframework.boot.test.mock.mockito.MockitoTestExecutionListener, org.springframework.boot.test.mock.mockito.ResetMocksTestExecutionListener, org.springframework.boot.test.autoconfigure.restdocs.RestDocsTestExecutionListener, (...)org.springframework.test.context.transaction.TransactionalTestExecutionListener, org.springframework.test.context.jdbc.SqlScriptsTestExecutionListener]

2018-02-08T10:58:42,448 INFO Using TestExecutionListeners: [org.springframework.test.context.web.ServletTestExecutionListener@f0ea28, org.springframework.test.context.support.DirtiesContextBeforeModesTestExecutionListener@16efaab,(...)org.springframework.boot.test.autoconfigure.web.servlet.MockMvcPrintOnlyOnFailureTestExecutionListener@9604d9]

2018-02-08T10:58:42,521 INFO Refreshing org.springframework.context.support.GenericApplicationContext@173f9fc: startup date [Thu Feb 08 10:58:42 IST 2018]; root of context hierarchy

2018-02-08T10:58:42,606 INFO JSR-330 'javax.inject.Inject' annotation found and supported for autowiring

2018-02-08T10:58:42,666 ERROR Caught exception while allowing TestExecutionListener [org.springframework.test.context.support.DependencyInjectionTestExecutionListener@19aaa5] to prepare test instance [io.github.thenilesh.service.impl.FooServiceTest@57f43]

org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'io.github.thenilesh.service.impl.FooServiceTest': Unsatisfied dependency expressed through field 'fooService'; nested exception is org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'io.github.thenilesh.service.FooService' available: expected at least 1 bean which qualifies as autowire candidate. Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true)}

at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredFieldElement.inject(AutowiredAnnotationBeanPostProcessor.java:588) ~[spring-beans-4.3.13.RELEASE.jar:4.3.13.RELEASE]

. . .

at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:192) [.cp/:?]

Caused by: org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'io.github.thenilesh.service.FooService' available: expected at least 1 bean which qualifies as autowire candidate. Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true)}

at org.springframework.beans.factory.support.DefaultListableBeanFactory.raiseNoMatchingBeanFound(DefaultListableBeanFactory.java:1493) ~[spring-beans-4.3.13.RELEASE.jar:4.3.13.RELEASE]

... 28 more

2018-02-08T10:58:42,698 INFO Closing org.springframework.context.support.GenericApplicationContext@173f9fc: startup date [Thu Feb 08 10:58:42 IST 2018]; root of context hierarchy

全栈跟踪

如果测试类使用@SpringBootTest进行注释,则它将创建整个应用程序上下文,包括数据库连接和许多不相关的bean,这些显然不是此单元测试所必需的(然后将不再是

单元 测试!)。可以预期的是,只有FooService依赖的bean 才能实例化,除了那些被模拟的bean 之外@MockBean

回答:

您应该使用@SpringBootTest(classes=FooServiceImpl.class)

正如在注释类型SpringBootTest上提到的:

公共抽象类[]类

带注释的类,用于加载ApplicationContext。也可以使用@ContextConfiguration(classes =

…)指定。如果未定义显式类,则测试将查找嵌套的@Configuration类,然后再返回SpringBootConfiguration搜索。

返回:用于加载应用程序上下文的带批注的类另请参见:ContextConfiguration.classes()

默认值:{}

这将仅加载必要的类。如果未指定,则可能会加载数据库配置和其他内容,从而使测试速度变慢。

另一方面,如果您真的想进行单元测试,则可以在不使用Spring的情况下测试此代码-

然后@RunWith(SpringRunner.class)@SpringBootTest就不需要注释了。您可以测试FooServiceImpl实例。如果您具有Autowired/

injected属性或服务,则可以通过setter,构造函数或Mockito模拟进行设置。

以上是 为什么组件扫描对Spring Boot单元测试不起作用? 的全部内容, 来源链接: utcz.com/qa/432815.html

回到顶部