it-swarm.com.ru

Java 8 извлекает первый ключ из соответствующего значения на карте

Предположим, у меня есть карта с указанным именем, парами фамилий, и я хочу найти имя первой записи на этой карте с фамилией, соответствующей определенному значению .. Как бы мы сделали это в стиле Java 8.

В моем тестовом примере ниже я поставил два способа сделать это.

Однако первый (ищущий данное имя первого человека с фамилией «Осел») выдаст Java.util.NoSuchElementException: значение отсутствует, поэтому это небезопасно.

Второй работает, но его не только сложнее читать, но и он не совсем функционален.

Просто интересно, предложит ли мне кто-нибудь из них более простой и ясный способ добиться этого, используя либо stream(), либо forEach(), либо и то, и другое.

@Test
public void shouldBeAbleToReturnTheKeyOfTheFirstMatchingValue() throws Exception {
    Map<String, String> names = new LinkedHashMap<>();
    names.put("John", "Doe");
    names.put("Fred", "Flintstone");
    names.put("Jane", "Doe");
    String keyOfTheFirst = names.entrySet().stream().filter(e -> e.getValue().equals("Doe")).findFirst().get().getKey();
    assertEquals("John", keyOfTheFirst);

    try {
        names.entrySet().stream().filter(e -> e.getValue().equals("Donkey")).findFirst().get();
    } catch (NoSuchElementException e){
        // Expected
    }

    Optional<Map.Entry<String, String>> optionalEntry = names.entrySet().stream().filter(e -> e.getValue().equals("Donkey")).findFirst();
    keyOfTheFirst = optionalEntry.isPresent() ? optionalEntry.get().getKey() : null;

    assertNull(keyOfTheFirst);
}

Заранее спасибо.

17
Julian

Чтобы вернуть значение по умолчанию, если совпадений нет, используйте Optional#orElse

names.entrySet().stream()
  .filter(e -> e.getValue().equals("Donkey"))
  .map(Map.Entry::getKey)
  .findFirst()
  .orElse(null);
52
Misha

Решение, предоставленное @Misha, является лучшим, если вы не хотите использовать сторонний код. Моя библиотека имеет специальный метод ярлыков ofKeys для тех случаев, когда я обнаружил, что это довольно распространенная задача:

StreamEx.ofKeys(names, "Donkey"::equals).findFirst().orElse(null);
1
Tagir Valeev

Из аналогичного вопроса :

public static <T, E> Set<T> getKeysByValue(Map<T, E> map, E value) {
    return map.entrySet()
              .stream()
              .filter(entry -> Objects.equals(entry.getValue(), value))
              .map(Map.Entry::getKey)
              .collect(Collectors.toSet());
}

Тогда вы можете выбрать первое, если хотите. Помните, что key уникален, value - нет.

Правка: Весь код (спасибо @Peter Lawrey)

package test;

import Java.util.LinkedHashMap;
import Java.util.Map;
import Java.util.Objects;
import Java.util.Optional;

public class Main {

    public static void main(String[] args) {
        Map<String, String> names = new LinkedHashMap<>();
        names.put("John", "Doe");
        names.put("Fred", "Flintstone");
        names.put("Jane", "Doe");

        Optional<String> firstKey = names.entrySet().stream()
                .filter(entry -> Objects.equals(entry.getValue(), "Doe"))
                .map(Map.Entry::getKey).findFirst();

        if (firstKey.isPresent()) {
            System.out.println(firstKey.get());
        }
    }
}
0
Doon

Ниже приведен мой фрагмент кода, чтобы получить ключ от карты, 

Map<String,String> pageDetails = new HashMap<String,String>();

public String getAssociatedKey(){
pageDetails.entrySet().stream().filter( e -> e.getValue().contains("John").findFirst().get().getKey();
}
0
Pavan T

Мне нравятся старомодные

static <K, V> K findFirstKeyByValue(Map<K, V> map, String value) {
    for (Entry<K, V> e : map.entrySet())
        if (e.getValue().equals(value))
            return e.getKey();
    return null;
}
0
saka1029