JMockit:bean 的模拟集合不适用于 Stream API
JMockit: mocked collection of beans doesn't work with Stream API
我有一个简单的方法需要使用 JMockit 进行测试,但似乎使用 Stream API 需要额外的模拟,但我不明白我到底需要添加什么。
使用for-loop
有效的方法
private final List<ValueProcessor> valueProcessors;
public List<Response> proc(Entry value) {
List<Response> badResponses = new ArrayList<>();
for (ValueProcessor processor : valueProcessors) {
List<Response> responses = processor.process(value);
if (!CollectionUtils.isEmpty(responses)) {
badResponses.addAll(responses);
}
}
return badResponses;
}
使用 Stream 的相同代码 API:
public List<Response> proc(Entry value) {
return valueProcessors.stream()
.map(processor -> processor.process(value))
.filter(collection -> !CollectionUtils.isEmpty(collection))
.flatMap(Collection::stream)
.collect(Collectors.toList());
}
我想用JMockit
测试代码,所以我有下面的测试
@RunWith(JMockit.class)
public class ProcessorTest {
@Injectable
private List<ValueProcessor> valueProcessors = Collections.singletonList(processor -> Collections.singletonList(new Response()));
@Tested
private Processor processor;
@Test
public void testProc() {
assertThat(processor.proc(null)).hasSize(1);
}
}
for-loop
方法的测试总是通过,Stream
方法总是失败。我该如何解决?
谢谢!
据我了解,当您使用 @Injectable
创建 valueProcessors
的列表时,您正在创建列表的模拟。
这意味着上述列表的方法将被模拟,以便它们默认为 return "empty" 值(如果未通过 Expectation
块指定任何内容)。
所以首先,非流代码,因为列表中没有调用任何方法,所以没有问题。
但是,在第二个使用流的代码中,对 valueProcessors.stream()
的调用执行 "empty" 模拟行为,即 return 是一个空流,因此,其他所有内容都为空也是。
我想出的解决方案是创建测试实例,并向其传递一个 "real" 列表,而不是模拟列表:
@RunWith(JMockit.class)
public class WhosebugTest {
// empty injectable just so that the processor can be created
@Injectable
private List<ValueProcessor> valueProcessors;
// availableDuringSetup = true to be able to modify the processor
@Tested(availableDuringSetup = true)
private Processor processor;
@Before
public void setUp() throws Exception {
// create new instance of the processor (now the valueProcessors list is a real list)
processor = new Processor(Collections.singletonList(processor -> Collections.singletonList(new Response())));
}
@Test
public void testProc() {
assertThat(processor.proc(null), hasSize(1));
}
}
我有一个简单的方法需要使用 JMockit 进行测试,但似乎使用 Stream API 需要额外的模拟,但我不明白我到底需要添加什么。
使用for-loop
private final List<ValueProcessor> valueProcessors;
public List<Response> proc(Entry value) {
List<Response> badResponses = new ArrayList<>();
for (ValueProcessor processor : valueProcessors) {
List<Response> responses = processor.process(value);
if (!CollectionUtils.isEmpty(responses)) {
badResponses.addAll(responses);
}
}
return badResponses;
}
使用 Stream 的相同代码 API:
public List<Response> proc(Entry value) {
return valueProcessors.stream()
.map(processor -> processor.process(value))
.filter(collection -> !CollectionUtils.isEmpty(collection))
.flatMap(Collection::stream)
.collect(Collectors.toList());
}
我想用JMockit
测试代码,所以我有下面的测试
@RunWith(JMockit.class)
public class ProcessorTest {
@Injectable
private List<ValueProcessor> valueProcessors = Collections.singletonList(processor -> Collections.singletonList(new Response()));
@Tested
private Processor processor;
@Test
public void testProc() {
assertThat(processor.proc(null)).hasSize(1);
}
}
for-loop
方法的测试总是通过,Stream
方法总是失败。我该如何解决?
谢谢!
据我了解,当您使用 @Injectable
创建 valueProcessors
的列表时,您正在创建列表的模拟。
这意味着上述列表的方法将被模拟,以便它们默认为 return "empty" 值(如果未通过 Expectation
块指定任何内容)。
所以首先,非流代码,因为列表中没有调用任何方法,所以没有问题。
但是,在第二个使用流的代码中,对 valueProcessors.stream()
的调用执行 "empty" 模拟行为,即 return 是一个空流,因此,其他所有内容都为空也是。
我想出的解决方案是创建测试实例,并向其传递一个 "real" 列表,而不是模拟列表:
@RunWith(JMockit.class)
public class WhosebugTest {
// empty injectable just so that the processor can be created
@Injectable
private List<ValueProcessor> valueProcessors;
// availableDuringSetup = true to be able to modify the processor
@Tested(availableDuringSetup = true)
private Processor processor;
@Before
public void setUp() throws Exception {
// create new instance of the processor (now the valueProcessors list is a real list)
processor = new Processor(Collections.singletonList(processor -> Collections.singletonList(new Response())));
}
@Test
public void testProc() {
assertThat(processor.proc(null), hasSize(1));
}
}