尽管创建了 2 个不同的模拟,但 InjectMocks 错误地将相同的模拟注入到 2 个类似类型的不同字段中

Sur*_*tra 5 java junit inject mockito maven

我有一个类,它有 2 个类似类型的字段。我嘲笑过他们两个。但是当我使用 InjectMocks 时,inject mocks 错误地将单个模拟注入到这两个字段中。

这是示例代码类:


import lombok.AccessLevel;
import lombok.RequiredArgsConstructor;

import java.util.Set;
import java.util.function.Consumer;

@RequiredArgsConstructor(access = AccessLevel.PRIVATE)
public class TestClass {

    private final Consumer<Set<Integer>> intConsumer;

    private final Consumer<Set<String>> stringConsumer;

    void PrintClass(){
        System.out.println("intConsumers: " + intConsumer);
        System.out.println("stringConsumers: " + stringConsumer);
    }
}
Run Code Online (Sandbox Code Playgroud)

这是测试类:


import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.InjectMocks;
import org.mockito.Mock;
import org.mockito.junit.MockitoJUnitRunner;

import java.util.Set;
import java.util.function.Consumer;

@RunWith(MockitoJUnitRunner.class)
public class TestClassTest {

    @Mock private Consumer<Set<Integer>> intConsumer;
    @Mock private Consumer<Set<String>> stringConsumer;
    @InjectMocks private TestClass testClass;

    @Test
    public void testPrint(){
        testClass.PrintClass();
    }


}
Run Code Online (Sandbox Code Playgroud)

这是我运行测试时的输出: testPrint() - intConsumer 被注入到 intConsumer 和 stringConsumer 中。

intConsumers: intConsumer
stringConsumers: intConsumer



Process finished with exit code 0
Run Code Online (Sandbox Code Playgroud)

我正在使用 Maven。

<dependency>
        <groupId>org.mockito</groupId>
        <artifactId>mockito-core</artifactId>
        <version>2.7.19</version>
</dependency>
Run Code Online (Sandbox Code Playgroud)

我创建了这个私有构造函数,专门用于使用 InjectMocks 进行测试。我不想将其设为 public/package-private 所以我不能使用字段注入。我也不想使用公共设置器公开这些字段。另外,我不想让我的领域成为非最终领域。

我已经尝试将 mockito-version 升级到 3.5.10,但它仍然有这个错误。我也尝试过使我的字段成为 final 并使用 setter - 然后注入工作正常 - 但我不想暴露我的 setter。我也试过用构造函数注入命名模拟 @Mock(name = "mock") 但它不起作用。

我在这里错过了什么吗?有没有办法让它与私有构造函数注入一起工作?

Les*_*iak 2

这是 Mockito 中的一个未解决的错误。

据我所知,PropertyAndSetterInjection 考虑了泛型类型和 @Mock 的名称属性,因此它按字段注入的预期工作。但它不适用于构造函数,因为 ConstructorInjection 仅使用 SimpleArgumentResolver ,这很好......非常简单,并且没有像属性注入器那样的任何 MockCandidateFilter 。

通常,您会:

  • 删除@InjectMocks并在测试的设置方法中构造TestClass实例。恕我直言,这是一种侵入性较小的方法。
  • 或者,如上所述,进行现场和入孵机注射工作。

这两种方法都不适用于您的约束(私有构造函数、无设置器、最终字段)。

在这种情况下,您可以诉诸反射来构建实例:

@Before
public void setUp() throws IllegalAccessException, 
        InvocationTargetException, 
        InstantiationException,
        NoSuchMethodException {
    final Constructor<TestClass> constructor = TestClass.class.getDeclaredConstructor(Consumer.class, Consumer.class);
    constructor.setAccessible(true);
    testClass = constructor.newInstance(intConsumer, stringConsumer);
}
Run Code Online (Sandbox Code Playgroud)