Java : List를 Map으로 변환하는 방법


224

최근에 나는 변환 할 수있는 최적의 방법이 될 것이다 대해 동료와 대화를 ListMap자바와 경우가 그렇게하는 특정 혜택을 제공합니다.

최적의 전환 접근 방식을 알고 싶습니다.

이 좋은 접근 방식입니까?

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

6
List는 개념적으로 Map과 다릅니다. Map은 '키, 값'쌍이라는 개념을 가지고 있지만 List는 그렇지 않습니다. 이것을 감안할 때 얼마나 정확하게 List에서 Map으로 변환하고 다시 변환할지 확실하지 않습니다.
Victor Sorokin

1
@Daniel : Optimal은 모든 방법 중 확실하지 않은 방법으로 목록을지도로 변환하는 몇 가지 방법을 보는 것이 좋을 것입니다.
Rachel


답변:


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()특정 매개 변수 로 설정해야 합니까?
Rachel

또한 맵의 가치는 무엇입니까? 예를 들어 자세히 설명 할 수 있습니까?
Rachel

1
@Rachel-값은 목록의 항목이며 키는 항목을 고유하게 만드는 것으로 사용자가 결정합니다. Jim의 사용 getKey()은 임의적이었습니다.
제레미

1
미리 크기를 알고 있으므로 Map <Key, Item> map = new HashMap <Key, Item> (list.size ());
Víctor Romero 2016 년

316

streamsCollectors클래스를 사용하여 한 줄에이를 수행 할 수 있습니다 .

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 -> itemi -> i

그리고 함수가 형용사가 아닌 경우 이진 연산자를 사용할 수 있습니다. 예를 들어 이것을 Listint 값의 경우 모듈로 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

1
@EmmanuelTouzery 음,를 Function.identity()반환합니다 t -> t;.
Alexis C.

2
물론 둘 다 작동합니다. 나는 그것이 맛의 문제라고 생각합니다. Function.identity ()가 더 즉시 인식 가능하다는 것을 알았습니다.
Emmanuel Touzery

OP는 어떤 pojo도 처리 할 수 ​​없으며 문자열과 정수만 처리 할 수 ​​없습니다 ::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
이 상단에 있어야합니다
마틴 앤더슨

6
" 구아바더 이상 사용되지 않는 이전 Google 컬렉션 라이브러리완전히 호환되는 상위 집합을 포함합니다 . 더 이상 해당 라이브러리를 사용하지 않아야합니다. "업데이트가 필요할 수 있습니다.
작은

3
이러한 간단한 작업을 위해 외부 라이브러리를 사용하는 것은 과도합니다. 그 또는 매우 약한 표준 라이브러리의 표시. 이 경우 @ jim-garrison의 대답은 완벽하게 합리적입니다. java에 "map"및 "reduce"와 같은 유용한 메소드가 없지만 완전히 필요한 것은 아닙니다.
linuxdan

2
구아바를 사용합니다. 불행히도 Guava는 Android에서 매우 느리기 때문에이 솔루션을 Android 프로젝트에서 사용해서는 안됩니다.
IgorGanapolsky

목록의 항목이 roleNames를 복제 한 경우 위 코드에서 예외가 발생합니다.
Junchen Liu

17

Java 8을 사용하면 다음을 수행 할 수 있습니다.

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

Value 사용하는 모든 개체가 될 수 있습니다.


16

Java 8부터 콜렉터를 사용한 @ZouZou대답Collectors.toMap 은 확실히이 문제를 해결하는 관용적 방법입니다.

그리고 이것은 일반적인 작업이므로 정적 유틸리티로 만들 수 있습니다.

그렇게하면 솔루션이 진정으로 하나의 라이너가됩니다.

/**
 * 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);

이 방법의 유형 매개 변수에 대한 자세한 내용은 다음 질문을 참조하십시오 .
glts

중복 키가있는 경우 예외가 발생합니다. Sth like : 스레드 "main"의 예외 java.lang.IllegalStateException : 중복 키 ... 자세한 내용은 다음을 참조하십시오. codecramp.com/java-8-streams-api-convert-list-map
EMM

@EMM 물론 Javadoc에 의도되고 문서화되어 있습니다.
16:11에

예, 중복 사례에 대한 답변을 업데이트했습니다. 검토하시기 바랍니다. 감사합니다
EMM

10

List과는 Map다른 개념이다. A List는 주문한 항목 모음입니다. 항목에는 중복 항목이 포함될 수 있으며 항목에는 고유 식별자 (키) 개념이 없을 수 있습니다. A Map에는 키에 매핑 된 값이 있습니다. 각 키는 하나의 값만 가리킬 수 있습니다.

따라서 귀하 List의 항목 에 따라 로 변환하거나 변환하지 못할 수 있습니다 Map. 님 List의 항목에 중복 항목이 없습니까? 각 항목에 고유 키가 있습니까? 그렇다면을에 넣을 수 있습니다 Map.


10

Alexis는 이미 method를 사용하여 Java 8에 답변을 게시했습니다 toMap(keyMapper, valueMapper). 이 메소드 구현에 대한 doc 에 따라 :

리턴 된 맵의 유형, 변경 가능성, 직렬화 가능성 또는 스레드 안전성에 대한 보장은 없습니다.

따라서 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 년에도 여전히 많은 사람들이 람다로 얻을 수있는 실제지도 구현을 알지 못한다는 재미있는 사실! 실제로 이것은 프로덕션에서 사용할 Java 8 람다에서 찾은 하나의 대답입니다.
Pavlo

병합 기능을 사용하지 않고 맵 유형을 지정하여 수집하는 방법이 있습니까?
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 멀티 맵 사용

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

모든 요소에 위치를 키로 제공 :

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

...

그것은 정말로 당신이 무엇을 원하느냐에 달려 있습니다.

예제에서 볼 수 있듯이 Map은 키에서 값으로의 매핑이며 목록은 각각 위치를 갖는 일련의 요소입니다. 따라서 단순히 자동으로 변환 할 수 없습니다.


그러나 List Element의 위치를 ​​키로 간주하고 그 가치를지도에 넣을 수 있습니다. 이것이 좋은 해결책입니까?
Rachel

AFAIK 예! JDK에는 자동으로 수행되는 기능이 없으므로 직접 롤백해야합니다.
다니엘

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

List<?>객체를 다음으로 변환하는 Java 8 예제 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>>>


1

Java-8 스트림 사용

results.stream().collect(Collectors.toMap(e->((Integer)e[0]), e->(String)e[1]));

0

Kango_V의 답변이 마음에 들지만 너무 복잡하다고 생각합니다. 나는 이것이 더 간단하다고 생각합니다-아마도 너무 간단합니다. 기울어지면 문자열을 일반 마커로 바꾸고 모든 키 유형에 사용할 수 있습니다.

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);
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.