Java:如何将列表转换为地图


224

最近我谈话有什么将要转换的最佳方式是同事List,以Map在Java中,如果有这样做的任何特殊利益。

我想知道最佳的转换方法,如果有人可以指导我,我将不胜感激。

这是个好方法吗?

List<Object[]> results;
Map<Integer, String> resultsMap = new HashMap<Integer, String>();
for (Object[] o : results) {
    resultsMap.put((Integer) o[0], (String) o[1]);
}

2
最佳的最佳方式是什么?优化时要牢记某些参数(速度/内存)。
Daniel Fath 2010年

6
List在概念上与Map不同-Map具有“键,值”对的概念,而List没有。鉴于此,目前尚不清楚您将如何从List转换到Map并返回。
维克多·索罗金

1
@Daniel:最佳,我的意思是在不确定所有方法之间的所有不同方法中,最好的方法是什么,因此最好能看到一些将列表转换为地图的不同方法。
雷切尔2010年


Answers:


188
List<Item> list;
Map<Key,Item> map = new HashMap<Key,Item>();
for (Item i : list) map.put(i.getKey(),i);

当然,假设每个Item都有一个getKey()返回正确类型的键的方法。


1
您也可以键入列表中的位置。
杰里米

@Jim:我需要将设置getKey()为任何特定参数吗?
雷切尔

另外Map中的值是多少,您可以举一个例子吗?
雷切尔2010年

1
@Rachel-值是列表中的项目,键是使项目唯一的东西,它由您确定。吉姆对的使用getKey()是任意的。
杰里米

1
您事先知道大小,因此可以执行Map <Key,Item> map = new HashMap <Key,Item>(list.size());
维克托·罗梅罗

316

,您就可以使用streamCollectors类在一行中完成此操作。

Map<String, Item> map = 
    list.stream().collect(Collectors.toMap(Item::getKey, item -> item));

简短的演示:

import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;

public class Test{
    public static void main (String [] args){
        List<Item> list = IntStream.rangeClosed(1, 4)
                                   .mapToObj(Item::new)
                                   .collect(Collectors.toList()); //[Item [i=1], Item [i=2], Item [i=3], Item [i=4]]

        Map<String, Item> map = 
            list.stream().collect(Collectors.toMap(Item::getKey, item -> item));

        map.forEach((k, v) -> System.out.println(k + " => " + v));
    }
}
class Item {

    private final int i;

    public Item(int i){
        this.i = i;
    }

    public String getKey(){
        return "Key-"+i;
    }

    @Override
    public String toString() {
        return "Item [i=" + i + "]";
    }
}

输出:

Key-1 => Item [i=1]
Key-2 => Item [i=2]
Key-3 => Item [i=3]
Key-4 => Item [i=4]

如评论中所述,您可以使用Function.identity()代替item -> item,尽管我发现i -> i很明确。

完整地请注意,如果函数不是双射的,则可以使用二进制运算符。例如,让我们考虑一下这个List函数和一个映射函数,该函数对于一个int值,以模3来计算其结果:

List<Integer> intList = Arrays.asList(1, 2, 3, 4, 5, 6);
Map<String, Integer> map = 
    intList.stream().collect(toMap(i -> String.valueOf(i % 3), i -> i));

运行此代码时,您会得到一条错误的提示java.lang.IllegalStateException: Duplicate key 1。这是因为1%3与4%3相同,因此在给定键映射功能的情况下具有相同的键值。在这种情况下,您可以提供合并运算符。

这是求和值的总和。(i1, i2) -> i1 + i2;可以用方法参考代替Integer::sum

Map<String, Integer> map = 
    intList.stream().collect(toMap(i -> String.valueOf(i % 3), 
                                   i -> i, 
                                   Integer::sum));

现在输出:

0 => 9 (i.e 3 + 6)
1 => 5 (i.e 1 + 4)
2 => 7 (i.e 2 + 5)

希望能帮助到你!:)


19
更好地使用,Function.identity()而不是item -> item
Emmanuel Touzery 2014年

1
@EmmanuelTouzery好吧,Function.identity()返回t -> t;
Alexis C.

2
当然,两者都可以。我想这是一个品味问题。我发现Function.identity()更容易识别。
Emmanuel Touzery 2015年

OP不处理任何pojos,仅处理您无法计算的字符串和整数::getKey
phil294

@Blauhirn我知道,我的示例基于下面的自定义类。您可以自由使用任何函数从值生成键。
Alexis C.

115

万一这个问题没有重复出现,正确的答案是使用Google收藏夹

Map<String,Role> mappedRoles = Maps.uniqueIndex(yourList, new Function<Role,String>() {
  public String apply(Role from) {
    return from.getName(); // or something else
  }});

17
它应该在顶部
Martin Andersson

6
Guava包含旧的,已弃用的Google Collections库的严格兼容超集。您不应再使用该库。 ”可能需要更新。
微小的

3
使用外部库进行这样的简单操作是过大的。那或者是非常弱的标准库的标志。在这种情况下,@ jim-garrison的答案是完全合理的。令人遗憾的是,java没有“ map”和“ reduce”之类的有用方法,但并非完全必要。
linuxdan

2
这使用番石榴。不幸的是,番石榴在Android上的速度非常慢,因此不应在Android项目中使用此解决方案。
IgorGanapolsky

如果在列表中的项目有重复的角色名,上面的代码会抛出异常,
君臣刘

17

使用Java 8,您可以执行以下操作:

Map<Key, Value> result= results
                       .stream()
                       .collect(Collectors.toMap(Value::getName,Function.identity()));

Value 可以是您使用的任何对象。


16

从Java 8开始, 使用Collectors.toMap收集器的@ZouZou回答肯定是解决此问题的惯用方式。

由于这是一项常见的任务,因此我们可以使其成为静态实用程序。

这样一来,解决方案便真正成为了一种选择。

/**
 * Returns a map where each entry is an item of {@code list} mapped by the
 * key produced by applying {@code mapper} to the item.
 *
 * @param list the list to map
 * @param mapper the function to produce the key from a list item
 * @return the resulting map
 * @throws IllegalStateException on duplicate key
 */
public static <K, T> Map<K, T> toMapBy(List<T> list,
        Function<? super T, ? extends K> mapper) {
    return list.stream().collect(Collectors.toMap(mapper, Function.identity()));
}

这是您将其用于的方式List<Student>

Map<Long, Student> studentsById = toMapBy(students, Student::getId);

有关此方法的类型参数的讨论,请参阅我的后续问题
2014年

万一有重复的键,这将引发异常。诸如此类:线程“ main”中的异常java.lang.IllegalStateException:重复的键....有关详细信息,请参见:codecramp.com/java-8-streams-api-convert-list-map
EMM

@EMM当然,正如Javadoc中所预期和记录的那样。
GLTS

是的,更新了涵盖重复案例的答案。请查阅。谢谢
EMM

10

一个ListMap在概念上是不同的。A List是项目的有序集合。这些项目可以包含重复项,并且一个项目可能没有任何有关唯一标识符(键)的概念。A Map具有映射到键的值。每个键只能指向一个值。

因此,根据您List的项目,可能无法将其转换为Map。您List的商品没有重复品吗?每个项目都有唯一的密钥吗?如果是这样,则可以将它们放入Map


10

Alexis已经使用method 在Java 8中发布了答案toMap(keyMapper, valueMapper)。根据此方法实现的文档

不能保证返回的Map的类型,可变性,可序列化性或线程安全性。

因此,如果我们对Map接口的特定实现感兴趣,HashMap那么可以将重载形式用作:

Map<String, Item> map2 =
                itemList.stream().collect(Collectors.toMap(Item::getKey, //key for map
                        Function.identity(),    // value for map
                        (o,n) -> o,             // merge function in case of conflict with keys
                        HashMap::new));         // map factory - we want HashMap and not any Map implementation

虽然无论是使用Function.identity()还是i->i不错,但它似乎Function.identity()不是i -> i可能会节省一些内存按照这个相关的答案


1
有趣的事实是,2019年仍有大量人仍未意识到他们不知道Lambda会带给他们真正的Map实现!实际上,这只是我在生产中将使用的Java 8 lambda找到的一个答案。
帕夫洛

是否可以通过指定Map类型但不使用合并功能来进行收集?
Rosberg Linhares


5

通用方法

public static <K, V> Map<K, V> listAsMap(Collection<V> sourceList, ListToMapConverter<K, V> converter) {
    Map<K, V> newMap = new HashMap<K, V>();
    for (V item : sourceList) {
        newMap.put( converter.getKey(item), item );
    }
    return newMap;
}

public static interface ListToMapConverter<K, V> {
    public K getKey(V item);
}

如何使用?我应该converter在方法中作为参数传递什么?
IgorGanapolsky

4

如果没有java-8,您将可以在一行Commons集合和Closure类中完成此操作

List<Item> list;
@SuppressWarnings("unchecked")
Map<Key, Item> map  = new HashMap<Key, Item>>(){{
    CollectionUtils.forAllDo(list, new Closure() {
        @Override
        public void execute(Object input) {
            Item item = (Item) input;
            put(i.getKey(), item);
        }
    });
}};

2

根据您要达到的目标,会想到许多解决方案:

每个列表项都是关键和价值

for( Object o : list ) {
    map.put(o,o);
}

列表元素可以查找一些内容,也许是名称:

for( MyObject o : list ) {
    map.put(o.name,o);
}

列表元素可以进行查找,并且不能保证它们是唯一的:使用Google的MultiMaps

for( MyObject o : list ) {
    multimap.put(o.name,o);
}

将所有元素的位置作为关键:

for( int i=0; i<list.size; i++ ) {
    map.put(i,list.get(i));
}

...

这实际上取决于您要达到的目标。

从示例中可以看到,映射是从键到值的映射,而列表只是一系列元素,每个元素都有一个位置。因此,它们根本无法自动转换。


但是我们可以将List Element的位置作为键并将其值放在map中,这是一个好的解决方案吗?
雷切尔

AFAIK是的!JDK中没有自动执行此功能的功能,因此您必须自己动手。
丹尼尔(Daniel)2010年

java 8流是否可以做最后一个版本(使用数组索引作为映射键)?
phil294

2

我正是为此目的编写了一个小方法。它使用来自Apache Commons的Validate。

随意使用它。

/**
 * Converts a <code>List</code> to a map. One of the methods of the list is called to retrive
 * the value of the key to be used and the object itself from the list entry is used as the
 * objct. An empty <code>Map</code> is returned upon null input.
 * Reflection is used to retrieve the key from the object instance and method name passed in.
 *
 * @param <K> The type of the key to be used in the map
 * @param <V> The type of value to be used in the map and the type of the elements in the
 *            collection
 * @param coll The collection to be converted.
 * @param keyType The class of key
 * @param valueType The class of the value
 * @param keyMethodName The method name to call on each instance in the collection to retrieve
 *            the key
 * @return A map of key to value instances
 * @throws IllegalArgumentException if any of the other paremeters are invalid.
 */
public static <K, V> Map<K, V> asMap(final java.util.Collection<V> coll,
        final Class<K> keyType,
        final Class<V> valueType,
        final String keyMethodName) {

    final HashMap<K, V> map = new HashMap<K, V>();
    Method method = null;

    if (isEmpty(coll)) return map;
    notNull(keyType, Messages.getString(KEY_TYPE_NOT_NULL));
    notNull(valueType, Messages.getString(VALUE_TYPE_NOT_NULL));
    notEmpty(keyMethodName, Messages.getString(KEY_METHOD_NAME_NOT_NULL));

    try {
        // return the Method to invoke to get the key for the map
        method = valueType.getMethod(keyMethodName);
    }
    catch (final NoSuchMethodException e) {
        final String message =
            String.format(
                    Messages.getString(METHOD_NOT_FOUND),
                    keyMethodName,
                    valueType);
        e.fillInStackTrace();
        logger.error(message, e);
        throw new IllegalArgumentException(message, e);
    }
    try {
        for (final V value : coll) {

            Object object;
            object = method.invoke(value);
            @SuppressWarnings("unchecked")
            final K key = (K) object;
            map.put(key, value);
        }
    }
    catch (final Exception e) {
        final String message =
            String.format(
                    Messages.getString(METHOD_CALL_FAILED),
                    method,
                    valueType);
        e.fillInStackTrace();
        logger.error(message, e);
        throw new IllegalArgumentException(message, e);
    }
    return map;
}

2

您可以利用Java 8的流API。

public class ListToMap {

  public static void main(String[] args) {
    List<User> items = Arrays.asList(new User("One"), new User("Two"), new User("Three"));

    Map<String, User> map = createHashMap(items);
    for(String key : map.keySet()) {
      System.out.println(key +" : "+map.get(key));
    }
  }

  public static Map<String, User> createHashMap(List<User> items) {
    Map<String, User> map = items.stream().collect(Collectors.toMap(User::getId, Function.identity()));
    return map;
  }
}

有关更多详细信息,请访问:http : //codecramp.com/java-8-streams-api-convert-list-map/


1

Java 8示例将List<?>对象的a 转换为Map<k, v>

List<Hosting> list = new ArrayList<>();
list.add(new Hosting(1, "liquidweb.com", new Date()));
list.add(new Hosting(2, "linode.com", new Date()));
list.add(new Hosting(3, "digitalocean.com", new Date()));

//example 1
Map<Integer, String> result1 = list.stream().collect(
    Collectors.toMap(Hosting::getId, Hosting::getName));

System.out.println("Result 1 : " + result1);

//example 2
Map<Integer, String> result2 = list.stream().collect(
    Collectors.toMap(x -> x.getId(), x -> x.getName()));

代码复制自:https :
//www.mkyong.com/java8/java-8-convert-list-to-map/


1

就像已经说过的,在Java-8中,Collectors提供了简洁的解决方案:

  list.stream().collect(
         groupingBy(Item::getKey)
        )

而且,您可以嵌套传递另一个groupingBy方法作为第二个参数的多个组:

  list.stream().collect(
         groupingBy(Item::getKey, groupingBy(Item::getOtherKey))
        )

这样,我们将获得多级地图,如下所示: Map<key, Map<key, List<Item>>>



0

我喜欢Kango_V的答案,但我认为它太复杂了。我认为这比较简单-也许太简单了。如果倾斜,则可以用通用标记替换String,并使它适用于任何Key类型。

public static <E> Map<String, E> convertListToMap(Collection<E> sourceList, ListToMapConverterInterface<E> converterInterface) {
    Map<String, E> newMap = new HashMap<String, E>();
    for( E item : sourceList ) {
        newMap.put( converterInterface.getKeyForItem( item ), item );
    }
    return newMap;
}

public interface ListToMapConverterInterface<E> {
    public String getKeyForItem(E item);
}

像这样使用:

        Map<String, PricingPlanAttribute> pricingPlanAttributeMap = convertListToMap( pricingPlanAttributeList,
                new ListToMapConverterInterface<PricingPlanAttribute>() {

                    @Override
                    public String getKeyForItem(PricingPlanAttribute item) {
                        return item.getFullName();
                    }
                } );

0

Apache Commons MapUtils.populateMap

如果您不使用Java 8,并且由于某种原因不想使用显式循环,请尝试MapUtils.populateMap使用Apache Commons。

MapUtils.populateMap

假设您有一个列表Pair

List<ImmutablePair<String, String>> pairs = ImmutableList.of(
    new ImmutablePair<>("A", "aaa"),
    new ImmutablePair<>("B", "bbb")
);

现在,您需要一个Pair的键映射到Pair对象。

Map<String, Pair<String, String>> map = new HashMap<>();
MapUtils.populateMap(map, pairs, new Transformer<Pair<String, String>, String>() {

  @Override
  public String transform(Pair<String, String> input) {
    return input.getKey();
  }
});

System.out.println(map);

给出输出:

{A=(A,aaa), B=(B,bbb)}

话虽如此,for循环可能更容易理解。(以下给出了相同的输出):

Map<String, Pair<String, String>> map = new HashMap<>();
for (Pair<String, String> pair : pairs) {
  map.put(pair.getKey(), pair);
}
System.out.println(map);
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.