使用Mockito,我如何匹配地图的键值对?

Sea*_*ean 14 java mockito

我需要根据特定的键值从模拟对象发送一个特定的值.

从具体类:

map.put("xpath", "PRICE");
search(map);
Run Code Online (Sandbox Code Playgroud)

从测试用例:

IOurXMLDocument mock = mock(IOurXMLDocument.class);
when(mock.search(.....need help here).thenReturn("$100.00");
Run Code Online (Sandbox Code Playgroud)

如何为此键值对模拟此方法调用?

Mar*_*uee 17

我发现这试图解决类似的问题,创建一个带有Map参数的Mockito存根.我不想为有问题的Map编写自定义匹配器,然后我找到了一个更优雅的解决方案:使用hamcrest-library中的其他匹配器与mockito的argThat:

when(mock.search(argThat(hasEntry("xpath", "PRICE"))).thenReturn("$100.00");
Run Code Online (Sandbox Code Playgroud)

如果您需要检查多个条目,那么您可以使用其他hamcrest好吃的东西:

when(mock.search(argThat(allOf(hasEntry("xpath", "PRICE"), hasEntry("otherKey", "otherValue")))).thenReturn("$100.00");
Run Code Online (Sandbox Code Playgroud)

对于非平凡的映射,这开始变得很长,所以我最终提取方法来收集条目匹配器并将它们粘贴在我们的TestUtils中:

import static org.hamcrest.Matchers.allOf;
import static org.hamcrest.Matchers.anyOf;
import static org.hamcrest.Matchers.hasEntry;

import java.util.ArrayList;
import java.util.List;
import java.util.Map;

import org.hamcrest.Matcher;
---------------------------------
public static <K, V> Matcher<Map<K, V>> matchesEntriesIn(Map<K, V> map) {
    return allOf(buildMatcherArray(map));
}

public static <K, V> Matcher<Map<K, V>> matchesAnyEntryIn(Map<K, V> map) {
    return anyOf(buildMatcherArray(map));
}

@SuppressWarnings("unchecked")
private static <K, V> Matcher<Map<? extends K, ? extends V>>[] buildMatcherArray(Map<K, V> map) {
    List<Matcher<Map<? extends K, ? extends V>>> entries = new ArrayList<Matcher<Map<? extends K, ? extends V>>>();
    for (K key : map.keySet()) {
        entries.add(hasEntry(key, map.get(key)));
    }
    return entries.toArray(new Matcher[entries.size()]);
}
Run Code Online (Sandbox Code Playgroud)

所以我离开了:

when(mock.search(argThat(matchesEntriesIn(map))).thenReturn("$100.00");
when(mock.search(argThat(matchesAnyEntryIn(map))).thenReturn("$100.00");
Run Code Online (Sandbox Code Playgroud)

有一些与泛型相关的丑陋,我正在压制一个警告,但至少它是干的并​​且隐藏在TestUtil中.

最后一点,请注意JUnit 4.10中嵌入式hamcrest问题.使用Maven,我建议首先导入hamcrest-library然后再导入JUnit 4.11(现在是4.12)并从JUnit中排除hamcrest-core只是为了测量:

<dependency>
    <groupId>org.hamcrest</groupId>
    <artifactId>hamcrest-library</artifactId>
    <version>1.3</version>
    <scope>test</scope>
</dependency>
<dependency>
    <groupId>junit</groupId>
    <artifactId>junit</artifactId>
    <version>4.12</version>
    <scope>test</scope>
    <exclusions>
        <exclusion>
            <groupId>org.hamcrest</groupId>
            <artifactId>hamcrest-core</artifactId>
        </exclusion>
    </exclusions>
</dependency>
<dependency>
    <groupId>org.mockito</groupId>
    <artifactId>mockito-all</artifactId>
    <version>1.9.5</version>
    <scope>test</scope>
</dependency>
Run Code Online (Sandbox Code Playgroud)

编辑:2017年9月1日 - 根据一些评论,我更新了我的答案,以显示我的Mockito依赖项,我在test util中的导入,以及从今天开始运行绿色的junit:

import static blah.tool.testutil.TestUtil.matchesAnyEntryIn;
import static blah.tool.testutil.TestUtil.matchesEntriesIn;
import static org.hamcrest.MatcherAssert.assertThat;
import static org.hamcrest.Matchers.is;
import static org.mockito.Matchers.argThat;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.when;

import java.util.HashMap;
import java.util.Map;

import org.junit.Test;

public class TestUtilTest {

    @Test
    public void test() {
        Map<Integer, String> expected = new HashMap<Integer, String>();
        expected.put(1, "One");
        expected.put(3, "Three");

        Map<Integer, String> actual = new HashMap<Integer, String>();
        actual.put(1, "One");
        actual.put(2, "Two");

        assertThat(actual, matchesAnyEntryIn(expected));

        expected.remove(3);
        expected.put(2, "Two");
        assertThat(actual, matchesEntriesIn(expected));
    }

    @Test
    public void mockitoTest() {
        SystemUnderTest sut = mock(SystemUnderTest.class);
        Map<Integer, String> expected = new HashMap<Integer, String>();
        expected.put(1, "One");
        expected.put(3, "Three");

        Map<Integer, String> actual = new HashMap<Integer, String>();
        actual.put(1, "One");

        when(sut.search(argThat(matchesAnyEntryIn(expected)))).thenReturn("Response");
        assertThat(sut.search(actual), is("Response"));
    }

    protected class SystemUnderTest {
        // We don't really care what this does
        public String search(Map<Integer, String> map) {
            if (map == null) return null;
            return map.get(0);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 这是一个方便的解决方案.奇怪的是,我得到了一个"错误的参数类型,期望Map <String,String>找到Map <?extends String,?extends String>",java似乎无法解决任何数量的prodding和import tweaking.我不得不改变我的模拟类的方法来接受`Map <?扩展String,?扩展String>`以使其工作.有什么想法吗?我和junit和hamcrest有相同的依赖版本; 我的Mockito是1.9.5. (2认同)

rog*_*ack 9

如果您只想与特定地图"匹配",您可以使用上面的一些答案,或者使用扩展Map的自定义"匹配器"或ArgumentCaptor,如下所示:

ArgumentCaptor<Map> argumentsCaptured = ArgumentCaptor.forClass(Map.class);
verify(mock, times(1)).method((Map<String, String>) argumentsCaptured.capture());
assert argumentsCaptured.getValue().containsKey("keyname"); 
// .getValue() will be the Map it called it with.
Run Code Online (Sandbox Code Playgroud)

另请参阅此处的更多答案:使用mockito验证对象属性值


Ray*_*Ray 8

对于像我这样提出这个问题的人来说,实际上有一个基于 Lambda 的非常简单的解决方案:

when(mock.search(argThat(map -> "PRICE".equals(map.get("xpath"))))).thenReturn("$100.00");
Run Code Online (Sandbox Code Playgroud)

说明:argThat需要一个ArgumentMatcher函数式接口,因此可以写成 Lambda。

  • 这是最简单、最优雅的方法。 (2认同)

Boz*_*zho 4

这行不通吗?

Map<String, String> map = new HashMap<String, String>();
map.put("xpath", "PRICE");
when(mock.search(map)).thenReturn("$100.00");
Run Code Online (Sandbox Code Playgroud)

Map参数的行为方式应与其他参数相同。