스트림에서 Collections.toMap ()을 사용할 때 목록의 반복 순서를 어떻게 유지합니까?


82

다음과 같이 Map에서 생성하고 List있습니다.

List<String> strings = Arrays.asList("a", "bb", "ccc");

Map<String, Integer> map = strings.stream()
    .collect(Collectors.toMap(Function.identity(), String::length));

에서와 동일한 반복 순서를 유지하고 싶습니다 List. 방법을 LinkedHashMap사용하여 어떻게 만들 수 Collectors.toMap()있습니까?


1
아래 내 대답을 확인하십시오. 그것은 정의를 사용하여 코드의 4 개 라인이다 Supplier, Accumulator그리고 Combiner에 대한 collect당신의 방법 stream:
hzitoun

1
질문에 대한 답변은 이미 완료되었으며,이 질문에 대한 답을 찾는 경로를 제시하고 싶습니다. (1)지도에서 순서를 원하고 LinkedHashMap을 사용해야합니다. 지도를 요청합니다. 따라서 Map이 필요한 곳에 LinkedHashMap을 사용하십시오.
Satyendra Kumar

답변:


116

2 개 매개 변수 버전은 다음을Collectors.toMap() 사용합니다 HashMap.

public static <T, K, U> Collector<T, ?, Map<K,U>> toMap(
    Function<? super T, ? extends K> keyMapper, 
    Function<? super T, ? extends U> valueMapper) 
{
    return toMap(keyMapper, valueMapper, throwingMerger(), HashMap::new);
}

4 개 매개 변수 버전 을 사용하려면 다음을 대체 할 수 있습니다.

Collectors.toMap(Function.identity(), String::length)

와:

Collectors.toMap(
    Function.identity(), 
    String::length, 
    (u, v) -> {
        throw new IllegalStateException(String.format("Duplicate key %s", u));
    }, 
    LinkedHashMap::new
)

또는 좀 더 깔끔하게 만들려면 새 toLinkedMap()메서드를 작성하고 다음을 사용하십시오.

public class MoreCollectors
{
    public static <T, K, U> Collector<T, ?, Map<K,U>> toLinkedMap(
        Function<? super T, ? extends K> keyMapper,
        Function<? super T, ? extends U> valueMapper)
    {
        return Collectors.toMap(
            keyMapper,
            valueMapper, 
            (u, v) -> {
                throw new IllegalStateException(String.format("Duplicate key %s", u));
            },
            LinkedHashMap::new
        );
    }
}

2
왜 그렇게 복잡합니까? 당신은 쉽게 할 수 있습니다, 아래 내 대답을 확인하십시오
hzitoun

1
mergeFunction4 매개 변수에 버전은 Collectors.toMap()키가 합병, 모두되고있는 대한 단서가없는 uv값이됩니다. 따라서 IllegalStateException에 대한 메시지는 옳지 않습니다.
MoonFruit

1
@hzitoun을 사용 Map::put하면 동일한 키에 대해 (아마도) 다른 값을 갖게되기 때문 입니다. 를 사용하여 Map::put처음 발견 한 값이 올바르지 않다고 간접적으로 선택했습니다. 이것이 최종 사용자가 원하는 것입니까? 확실합니까? 그렇다면 Map::put. 그렇지 않으면 확실하지 않고 결정할 수 없습니다. 스트림이 값이 다른 두 개의 동일한 키에 매핑되었음을 사용자에게 알립니다. 나는 당신 자신의 대답에 대해 언급했을 것이지만 현재 잠겨 있습니다.
Olivier Grégoire

69

자신의 Supplier, AccumulatorCombiner:

    List<String> myList = Arrays.asList("a", "bb", "ccc"); 
    // or since java 9 List.of("a", "bb", "ccc");
    
    LinkedHashMap<String, Integer> mapInOrder = myList
        .stream()
        .collect(
            LinkedHashMap::new,                                   // Supplier
            (map, item) -> map.put(item, item.length()),          // Accumulator
            Map::putAll);                                         // Combiner

    System.out.println(mapInOrder);  // {a=1, bb=2, ccc=3}

1
@Sushil 수락 된 답변은 논리를 재사용 할 수 있습니다
cylinder.y

1
스트림은 map.put(..)잘못된 것과 같은 부작용에 의존합니다 .
Nikolas Charalambidis는

무엇이 더 빠른지 아십니까? 귀하의 버전 또는 허용 된 답변을 사용하고 계십니까?
nimo23

@Nikolas 부작용에 대해 설명해 주 시겠습니까? 실제로 어떤 것을 선택할지 결정하는 데 문제가 있습니다. stackoverflow.com/questions/61479650/…
nimo23

0

Kotlin에서는 toMap()주문이 보존됩니다.

fun <K, V> Iterable<Pair<K, V>>.toMap(): Map<K, V>

주어진 쌍 모음에서 모든 키-값 쌍을 포함하는 새 맵을 반환합니다.

반환 된 맵은 원래 컬렉션의 항목 반복 순서를 유지합니다. 두 쌍 중 하나가 동일한 키를 갖는 경우 마지막 하나가 맵에 추가됩니다.

구현은 다음과 같습니다.

public fun <K, V> Iterable<Pair<K, V>>.toMap(): Map<K, V> {
    if (this is Collection) {
        return when (size) {
            0 -> emptyMap()
            1 -> mapOf(if (this is List) this[0] else iterator().next())
            else -> toMap(LinkedHashMap<K, V>(mapCapacity(size)))
        }
    }
    return toMap(LinkedHashMap<K, V>()).optimizeReadOnlyMap()
}

사용법은 간단합니다.

val strings = listOf("a", "bb", "ccc")
val map = strings.map { it to it.length }.toMap()

의 기본 컬렉션 mapLinkedHashMap(삽입 순서)입니다.


0

일부 필드로 객체 배열을 매핑하는 간단한 함수 :

public static <T, E> Map<E, T> toLinkedHashMap(List<T> list, Function<T, E> someFunction) {
    return list.stream()
               .collect(Collectors.toMap(
                   someFunction, 
                   myObject -> myObject, 
                   (key1, key2) -> key1, 
                   LinkedHashMap::new)
               );
}


Map<String, MyObject> myObjectsByIdMap1 = toLinkedHashMap(
                listOfMyObjects, 
                MyObject::getSomeStringField()
);

Map<Integer, MyObject> myObjectsByIdMap2 = toLinkedHashMap(
                listOfMyObjects, 
                MyObject::getSomeIntegerField()
);
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.