14

我有一个对我来说似乎很奇怪的问题。我有以下设置:

一个接口:

package com.example;

public interface SomeDependency {
}

一个弹簧组件:

package com.example;

@Component
public class SomeClass {
}

带有由 EasyMock 生成的模拟 bean 的弹簧测试配置:

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

    <bean id="someInterfaceMock" class="org.easymock.EasyMock" factory-method="createMock">
        <constructor-arg value="com.example.SomeDependency" />
    </bean> 
</beans>

还有一个单元测试:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("/testconfig.xml")
public class SomeClassTest {

    @Autowired
    SomeClass someClass;

    @Autowired
    SomeDependency someDependency;

    @Test
    public void testSomeClass() throws Exception {
        assertNotNull(someClass);
    }

    @Test
    public void testSomeDependency() throws Exception {
        assertNotNull(someDependency);
    }
}

项目编译并且测试通过没有任何问题,即SomeClass(一个“真实”对象)和SomeDependency(一个由 EasyMock 生成的模拟对象)的自动装配成功。

但是,如果我将SomeClass的实现更改为:

@Component
public class SomeClass {

    @Autowired
    SomeDependency someDependency;
}

两个测试都失败了,因为

Caused by: org.springframework.beans.factory.NoSuchBeanDefinitionException: No matching bean of type [com.example.SomeDependency] found for dependency: expected at least 1 bean which qualifies as autowire candidate for this dependency. Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true)}

所以我的问题是:

  1. 为什么 Spring 无法将依赖项自动连接到SomeClass(当它成功地将相同的依赖项自动连接到SomeClassTest时)?
  2. 如何更改SomeClassTesttestconfig.xml以使测试通过?

评论:实际上SomeClass所代表的类是框架的一部分。因此,它不能轻易更新,至少不能在合理的时间内更新。

依赖项:

  • 春天:3.0.5.RELEASE
  • 易模拟:3.0

编辑:

从 Spring 3.2 RC1 开始,解决了泛型工厂方法和模拟对象的问题

/马蒂亚斯

4

1 回答 1

23

在使用工厂创建具有自动装配功能的 bean 时,xml 中定义的顺序似乎实际上很重要。如果您放置someInterfaceMock上面的声明,component-scan它将起作用。

一些说明原因:当 Spring 尝试自动装配SomeClass时,它会搜索 type 的 bean SomeDependency。在这个阶段someInterfaceMock仍然是一个工厂,所以 Spring 检查EasyMock.createMock(...)返回的工厂方法的签名,<T>所以 Spring 只找到一个Object不是所需类型的。

更好的方法是使用 Spring 的FactoryBean接口来创建你的模拟。

这是一个应该工作的基本实现:

public class EasyMockFactoryBean<T> implements FactoryBean<T> {
    private Class<T> mockedClass;

    public void setMockedClass(Class mockedClass) {
        this.mockedClass = mockedClass;
    } 

    public T getObject() throws Exception {
        return EasyMock.createMock(mockedClass);
    }

    public Class<T> getObjectType() {
        return mockedClass;
    }

    public boolean isSingleton() {
        return true;
    } 

}

这是 bean 定义(顺序无关紧要!):

<bean class="com.example.EasyMockFactoryBean">
    <property name="mockedClass" value="com.example.Dependancy"/>
</bean>    
于 2011-06-14T07:28:27.633 回答