我有一个类,它有 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") 但它不起作用。
我在这里遗漏了什么吗?有没有办法让它与私有构造函数注入一起工作?
这是 Mockito 中的一个未解决的错误。
据我所知,PropertyAndSetterInjection 考虑了泛型类型和 @Mock 的名称属性,因此它按字段注入的预期工作。但它不适用于构造函数,因为 ConstructorInjection 仅使用 SimpleArgumentResolver,这很好......非常简单,并且没有像属性注入器那样的任何 MockCandidateFilter 。
通常,您会:
这两种方法都不适用于您的约束(私有构造函数、无设置器、最终字段)。
在这种情况下,您可以诉诸反射来构建实例:
@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);
}
您可以使用反射:
@BeforeEach
void setup() {
ReflectionTestUtils.setField(testClass, "intConsumer", intConsumer);
ReflectionTestUtils.setField(testClass, "stringConsumer", throttleAndBulkTrackService);
}