【问题标题】:Spring 3 and JUnit 4 (autowiring)Spring 3 和 JUnit 4(自动装配)
【发布时间】:2012-08-13 18:06:22
【问题描述】:

我是 Spring MVC 和 JUnit 的新手。基本上我想自动装配服务类,这个类应该在spring上下文中加载。

服务

@服务 公共类 FundService { @自动连线 FundDAO /** * @返回 */ 公共列表 getFundDetails(String productId) { return fundDAO.getFundDetails(productId); } }

应用程序上下文

<beans>

    <mvc:annotation-driven />

    <context:component-scan base-package="com.test.*" />
</beans>

Junit 类

@RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration(locations={"classpath*:/WEB-INF/application-context.xml"}) 公共类 CompensationServiceTest { @自动连线 私人基金服务基金服务; @测试 公共无效验证GetCompensationList() { System.out.println(fundService == null); } }

在执行测试时,我得到以下异常跟踪

org.springframework.beans.factory.BeanCreationException:创建名为“com.test.admin.service.CompensationServiceTest”的bean时出错:注入自动装配的依赖项失败;嵌套异常是 org.springframework.beans.factory.BeanCreationException:无法自动装配字段:私有 com.test.admin.service.FundService com.test.admin.service.CompensationServiceTest.fundService;嵌套异常是 org.springframework.beans.factory.NoSuchBeanDefinitionException:没有为依赖项找到类型为 [com.test.admin.service.FundService] 的匹配 bean:预计至少有 1 个 bean 有资格作为此依赖项的自动装配候选者。依赖注解:{@org.springframework.beans.factory.annotation.Autowired(required=true)} 在 org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor.postProcessPropertyValues(AutowiredAnnotationBeanPostProcessor.java:286) 在 org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.populateBean(AbstractAutowireCapableBeanFactory.java:1064) 在 org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.autowireBeanProperties(AbstractAutowireCapableBeanFactory.java:374) 在 org.springframework.test.context.support.DependencyInjectionTestExecutionListener.injectDependencies(DependencyInjectionTestExecutionListener.java:110) 在 org.springframework.test.context.support.DependencyInjectionTestExecutionListener.prepareTestInstance(DependencyInjectionTestExecutionListener.java:75) 在 org.springframework.test.context.TestContextManager.prepareTestInstance(TestContextManager.java:333) 在 org.springframework.test.context.junit4.SpringJUnit4ClassRunner.createTest(SpringJUnit4ClassRunner.java:220) 在 org.springframework.test.context.junit4.SpringJUnit4ClassRunner$1.runReflectiveCall(SpringJUnit4ClassRunner.java:301) 在 org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15) 在 org.springframework.test.context.junit4.SpringJUnit4ClassRunner.methodBlock(SpringJUnit4ClassRunner.java:303) 在 org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:240) 在 org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:49) 在 org.junit.runners.ParentRunner$3.run(ParentRunner.java:193) 在 org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:52) 在 org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191) 在 org.junit.runners.ParentRunner.access$000(ParentRunner.java:42) 在 org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:184) 在 org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61) 在 org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:70) 在 org.junit.runners.ParentRunner.run(ParentRunner.java:236) 在 org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:180) 在 org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:49) 在 org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38) 在 org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467) 在 org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683) 在 org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390) 在 org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197) 引起:org.springframework.beans.factory.BeanCreationException:无法自动装配字段:private com.test.admin.service.FundService com.test.admin.service.CompensationServiceTest.fundService;嵌套异常是 org.springframework.beans.factory.NoSuchBeanDefinitionException:没有为依赖项找到类型为 [com.test.admin.service.FundService] 的匹配 bean:预计至少有 1 个 bean 有资格作为此依赖项的自动装配候选者。依赖注解:{@org.springframework.beans.factory.annotation.Autowired(required=true)} 在 org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredFieldElement.inject(AutowiredAnnotationBeanPostProcessor.java:507) 在 org.springframework.beans.factory.annotation.InjectionMetadata.inject(InjectionMetadata.java:84) 在 org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor.postProcessPropertyValues(AutowiredAnnotationBeanPostProcessor.java:283) ... 26 更多 原因:org.springframework.beans.factory.NoSuchBeanDefinitionException:没有为依赖项找到类型为 [com.test.admin.service.FundService] 的匹配 bean:预计至少有 1 个 bean 有资格作为此依赖项的自动装配候选者。依赖注解:{@org.springframework.beans.factory.annotation.Autowired(required=true)} 在 org.springframework.beans.factory.support.DefaultListableBeanFactory.raiseNoSuchBeanDefinitionException(DefaultListableBeanFactory.java:903) 在 org.springframework.beans.factory.support.DefaultListableBeanFactory.doResolveDependency(DefaultListableBeanFactory.java:772) 在 org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveDependency(DefaultListableBeanFactory.java:686) 在 org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredFieldElement.inject(AutowiredAnnotationBeanPostProcessor.java:478) ... 28 更多

【问题讨论】:

    标签: spring junit4 autowired


    【解决方案1】:

    首先,WEB-INF 文件夹不应该在类路径中;它应该在项目的文件系统中。因此,如果您使用 Maven 项目结构,该文件夹将相对于您的项目的根目录(即src/main/webapp/WEB-INF)。在这种情况下,您可能希望将该文件夹中的 XML 配置文件声明为文件系统资源,如下所示:

    @ContextConfiguration("file:src/main/webapp/WEB-INF/application-context.xml")

    其次,如果这是一个测试配置文件,则不应将其存储在WEB-INF 下。相反,您应该将其存储在类路径中。按照 Maven 项目结构,这将是 src/test/resources/application-context.xml,在这种情况下,您将在测试中使用以下声明:

    @ContextConfiguration("/application-context.xml")

    @ContextConfiguration("classpath:application-context.xml")

    ...无论您喜欢哪个,但请注意它们是等价的。

    问候和感谢,

    山姆

    【讨论】:

      【解决方案2】:

      有两种选择:

      如果你使用@Autowire,这意味着你正在为你的测试创建一个应用程序上下文,有几个步骤可以实现这一点;我不想告诉你怎么做,而是告诉你正确的方法。

      既然您说的是单元测试,这意味着您正在测试的类是隔离的,并且注入的依赖项必须是模拟的,下一个示例:

      班级:

      @RestController
      @RequestMapping(value = "/")
      public class CatalogRest {
      
          @Autowired
          private CatalogService catalogService;
      
          @Autowired
          private LoggedUser loggedUser;
      
          @RequestMapping(value = GET_VIEW_TYPE, method = POST)
          public ResponseEntity<List<ViewType>> getViewType() {
      
              List<ViewType> viewTypes = catalogService.getViewType(loggedUser.getUserId());
              return new ResponseEntity<List<ViewType>>(viewTypes, HttpStatus.OK);
      
          }
      

      现在开始测试:

       @RunWith(MockitoJUnitRunner.class)
      public class CatalogRestTest {
      
          @InjectMocks
          private CatalogRest subject;
      
          @Mock
          private CatalogService catalogService;
      
          @Mock
          private LoggedUser loggedUser;
      
          @Before
          public void init() {
              MockitoAnnotations.initMocks(this);
          }
      
          @Test
          public void classAndItsMethodsHaveRequiredAnnotations() throws Exception {
              assertTrue(subject.getClass().isAnnotationPresent(RestController.class));
              assertTrue(subject.getClass().isAnnotationPresent(RequestMapping.class));
      
              Method getViewTypeMethod = subject.getClass().getMethod("getViewType");
              getViewTypeMethod.isAnnotationPresent(RequestMapping.class);
          }
      
          @Test
          public void getViewType_whenInvoked_returnsAListOfViewTypes() throws Exception {
              List<ViewType> viewTypes = Arrays.asList(new ViewType());
      
              when(loggedUser.getUserId()).thenReturn("fake user");
              when(catalogService.getViewType(eq("fake user"))).thenReturn(viewTypes);
      
              ResponseEntity<List<ViewType>> result = subject.getViewType();
      
              assertNotNull(result);
              assertNotNull(result.getBody());
              assertTrue(result.getBody().size() == 1);
      
              verify(catalogService, times(1)).getViewType(eq("fake user"));
          }
      

      不要尝试使用 spring 进行单元测试,它应该只用于integration testing

      【讨论】:

        【解决方案3】:

        您必须使用“@Component”注释限定服务类才能在组件扫描期间被选中,对吧?

        【讨论】:

        • @Service 也做同样的事情。与组件一样,它也是立体类型之一。
        【解决方案4】:

        你能把你做bean配置的代码贴出来吗?这里的问题是 Spring 试图找到一个具体的类来连接

        @Autowired
        private FundService fundService;
        

        您需要指定 Spring 应该在您的配置中注入的 bean。例如,在 XML 中执行它你会说:

        <beans>
        
            <mvc:annotation-driven />
            <context:component-scan base-package="com.test.*" />
        
            <!-- Assuming FundServiceImpl is a class that implement FundService -->
            <bean id="fundService" class="com.test.admin.service.FundServiceImpl"/> 
        </beans>
        

        【讨论】:

          【解决方案5】:

          就我而言,我有嵌套的分类/参数化测试:

          之前

          @RunWith(Enclosed.class)
          @ContextConfiguration(classes = {TestX.class})
          public class SimpleTest {
          
          ...
          @RunWith(Parameterized.class)
          public static class SimpleCheckAddTest{
          .....
          }
          
          @RunWith(Parameterized.class)
          public static class SimpleCheckRemoveTest{
          .....
          }
          
          }
          

          得到:<.....unsatisfied dependency express through field......> 类似异常。

          之后

          @RunWith(Enclosed.class)
          public class SimpleTest {
          
          ...
          @RunWith(Parameterized.class)
          @ContextConfiguration(classes = {TestX.class})
          public static class SimpleCheckAddTest{
          .....
          }
          
          @RunWith(Parameterized.class)
          @ContextConfiguration(classes = {TestX.class})
          public static class SimpleCheckRemoveTest{
          .....
          }
          
          }
          

          【讨论】:

            猜你喜欢
            • 2011-04-09
            • 1970-01-01
            • 1970-01-01
            • 1970-01-01
            • 2011-06-10
            • 1970-01-01
            • 1970-01-01
            • 1970-01-01
            • 2018-12-01
            相关资源
            最近更新 更多