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

问题描述 投票:0回答:2

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

这里是示例代码类:


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);
    }
}

这是测试类:


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();
    }


}

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

intConsumers: intConsumer
stringConsumers: intConsumer



Process finished with exit code 0

我正在使用 Maven。

<dependency>
        <groupId>org.mockito</groupId>
        <artifactId>mockito-core</artifactId>
        <version>2.7.19</version>
</dependency>

我创建了这个私有构造函数,专门用于使用 InjectMocks 进行测试。我不想将其设为公共/包私有,因此我无法使用字段注入。我也不想使用公共设置器公开这些字段。另外,我不想让我的字段成为非最终字段。

我已尝试将mockito版本升级到3.5.10,但仍然存在此错误。 我还尝试将我的字段设为最终字段并使用设置器 - 然后注入工作正常 - 但我不想暴露我的设置器。 我还尝试使用构造函数注入命名模拟 @Mock(name = "mock") 但它不起作用。

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

java maven junit mockito inject
2个回答
3
投票

这是 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);
}

0
投票

您可以使用反射:

@BeforeEach
void setup() {
    ReflectionTestUtils.setField(testClass, "intConsumer", intConsumer);
    ReflectionTestUtils.setField(testClass, "stringConsumer", throttleAndBulkTrackService);
}
© www.soinside.com 2019 - 2024. All rights reserved.