будівельник для HashMap


Відповіді:


20

Оскільки Mapінтерфейс Java 9 містить:

  • Map.of(k1,v1, k2,v2, ..)
  • Map.ofEntries(Map.entry(k1,v1), Map.entry(k2,v2), ..).

Обмеженнями цих заводських методів є те, що вони:

  • не може містити nulls як ключі та / або значення (якщо вам потрібно зберегти нулі, подивіться інші відповіді)
  • виготовити незмінні карти

Якщо нам потрібна змінна карта (на зразок HashMap), ми можемо використовувати її конструктор копій і дозволити копіювати вміст карти, створеної за допомогоюMap.of(..)

Map<Integer, String> map = new HashMap<>( Map.of(1,"a", 2,"b", 3,"c") );

2
Зауважте, що методи Java 9 не допускають nullзначень, що може бути проблемою залежно від випадку використання.
Пер Лундберг

@JoshM. ІМО Map.of(k1,v1, k2,v2, ...)можна використовувати безпечно, коли у нас не так багато значень. Більш велика кількість значень Map.ofEntries(Map.entry(k1,v1), Map.entry(k2,v2), ...)дає нам більше читабельний код, який менш схильний до помилок (якщо я вас неправильно зрозумів).
Пшемо

Ви прекрасно зрозуміли. Перший для мене просто грубий; Я відмовляюся його використовувати!
Джош М.

164

Для HashMaps такого немає, але ви можете створити ImmutableMap за допомогою будівельника:

final Map<String, Integer> m = ImmutableMap.<String, Integer>builder().
      put("a", 1).
      put("b", 2).
      build();

А якщо вам потрібна змінна карта, ви можете просто подати її конструктору HashMap.

final Map<String, Integer> m = Maps.newHashMap(
    ImmutableMap.<String, Integer>builder().
        put("a", 1).
        put("b", 2).
        build());

43
ImmutableMapне підтримує nullзначення. Таким чином , існує обмеження такого підходу: ви не можете встановити значення в вашому HashMapTo null.
vitaly

5
sean-patrick-floyd Ну, один практичний приклад: Spring's NamedParameterJdbcTemplate очікує карту значень , введених іменами параметрів. Скажімо, я хочу використовувати NamedParameterJdbcTemplate для встановлення нульового значення стовпця. Я не бачу: а) як це кодовий запах; б) як тут використовувати нульовий шаблон об’єкта
vitaly

2
@vitaly не може з цим посперечатися
Шон Патрік Флойд

2
Чи є щось не так у використанні new HashMapконструктора Java замість статичного Maps.newHashMapметоду?
CorayThan

1
@CorayThan - Джонік правильний, це лише ярлик, який покладається на статичний висновок. stackoverflow.com/a/13153812
AndersDJohnson

46

Не зовсім конструктор, але використовуючи ініціалізатор:

Map<String, String> map = new HashMap<String, String>() {{
    put("a", "1");
    put("b", "2");
}};

Зачекайте. Хіба це не зробить map instanceof HashMapпомилковим? Схоже на не дуже велику ідею.
Елазар Лейбович

3
@Elazar map.getClass()==HashMap.classповерне помилковий. Але це все-таки дурне випробування. HashMap.class.isInstance(map)слід віддати перевагу, і це повернеться правдою.
Шон Патрік Флойд

59
Це сказало: Я все ще думаю, що це рішення є злим.
Шон Патрік Флойд

11
Це ініціалізатор екземпляра, а не статичний ініціалізатор. Він запускається після конструктора супер, але перед тілом конструктора, для кожного конструктора в класі. Життєвий цикл не дуже відомий, тому я уникаю цієї ідіоми.
Джо Кодер

14
Це дуже погане рішення , і слід уникати: stackoverflow.com/a/27521360/3253277
Alexandre Дюбрейля

36

Це схоже на прийняту відповідь, але трохи чіткіше, на мій погляд:

ImmutableMap.of("key1", val1, "key2", val2, "key3", val3);

Вищеописаного методу існує кілька варіантів, і вони чудово підходять для створення статичних, незмінних, незмінних карт.


4
Я попросив будівельника. Ви обмежені кількома елементами.
Елазар Лейбович

Хороший і чистий, але все-таки змушує мене шукати оператора Perl => ... що дивно.
Аарон Маенпаа

10

Ось дуже простий ...

public class FluentHashMap<K, V> extends java.util.HashMap<K, V> {
  public FluentHashMap<K, V> with(K key, V value) {
    put(key, value);
    return this;
  }

  public static <K, V> FluentHashMap<K, V> map(K key, V value) {
    return new FluentHashMap<K, V>().with(key, value);
  }
}

тоді

import static FluentHashMap.map;

HashMap<String, Integer> m = map("a", 1).with("b", 2);

Дивіться https://gist.github.com/culmat/a3bcc646fa4401641ac6eb01f3719065


Мені подобається простота вашого підходу. Тим більше, що це 2017 рік (вже майже 2018 рік!), А такого API ще немає в JDK
Milad Naseri

Дуже здорово, спасибі. @MiladNaseri Це божевільно, що в JDK все ще немає подібного в своєму API, яка страшна ганьба.
неймовірне

9

Простий конструктор карт тривіально написати:

public class Maps {

    public static <Q,W> MapWrapper<Q,W> map(Q q, W w) {
        return new MapWrapper<Q, W>(q, w);
    }

    public static final class MapWrapper<Q,W> {
        private final HashMap<Q,W> map;
        public MapWrapper(Q q, W w) {
            map = new HashMap<Q, W>();
            map.put(q, w);
        }
        public MapWrapper<Q,W> map(Q q, W w) {
            map.put(q, w);
            return this;
        }
        public Map<Q,W> getMap() {
            return map;
        }
    }

    public static void main(String[] args) {
        Map<String, Integer> map = Maps.map("one", 1).map("two", 2).map("three", 3).getMap();
        for (Map.Entry<String, Integer> entry : map.entrySet()) {
            System.out.println(entry.getKey() + " = " + entry.getValue());
        }
    }
}

6

Ви можете використовувати:

HashMap<String,Integer> m = Maps.newHashMap(
    ImmutableMap.of("a",1,"b",2)
);

Це не так класно і читабельно, але робить роботу.


1
Map<String, Integer> map = ImmutableMap.of("a", 1, "b", 2);, краще?
lschin

Те саме, що і конструктор, але з обмеженою кількістю даних, оскільки він реалізований з перевантаженнями. Якщо у вас є лише кілька елементів - я думаю, що це краще.
Елазар Лейбович

4

HashMapє змінним; немає потреби в будівельнику.

Map<String, Integer> map = Maps.newHashMap();
map.put("a", 1);
map.put("b", 2);

Що робити, якщо ви хочете ініціалізувати поле з ним? Вся логіка в одному рядку краща за логіку, розкидану між полем і c'tor.
Елазар Лейбович

@Elazar: Якщо ви хочете ініціалізувати поле з конкретними значеннями, які відомі в такий час компіляції, зазвичай ви хочете, щоб це поле було незмінним і його слід використовувати ImmutableSet. Якщо ви дійсно хочете, щоб він мінявся, ви можете ініціалізувати його в конструкторі або блоці ініціалізатора екземпляра або блоці статичного ініціалізатора, якщо це статичне поле.
ColinD

1
Е, мав би сказати ImmutableMapтам очевидно.
ColinD

Я не думаю, що так. Я вважаю за краще інтіалізацію в тому ж рядку визначення, а потім затягнути їх у нестатичну інтіалізацію {{init();}}(не в конструктор, оскільки інший конструктор може це забути). І приємно, що це свого роду атомна дія. Якщо карта є нестабільною, тоді, використовуючи її за допомогою конструктора, переконайтесь, що вона завжди є nullабо в остаточному стані, ніколи не заповнена наполовину.
Елазар Лейбович

1

Ви можете безперебійно використовувати API в колекціях Eclipse :

Map<String, Integer> map = Maps.mutable.<String, Integer>empty()
        .withKeyValue("a", 1)
        .withKeyValue("b", 2);

Assert.assertEquals(Maps.mutable.with("a", 1, "b", 2), map);

Ось блог з більш детальною інформацією та прикладами.

Примітка. Я є членом колекції Eclipse.


0

У мене була така ж вимога деякий час назад. Це не має нічого спільного з Guava, але ви можете зробити щось подібне, щоб мати можливість чисто сконструювати, Mapвикористовуючи вільний конструктор .

Створіть базовий клас, який розширює Map.

public class FluentHashMap<K, V> extends LinkedHashMap<K, V> {
    private static final long serialVersionUID = 4857340227048063855L;

    public FluentHashMap() {}

    public FluentHashMap<K, V> delete(Object key) {
        this.remove(key);
        return this;
    }
}

Потім створіть вільний конструктор методами, які відповідають вашим потребам:

public class ValueMap extends FluentHashMap<String, Object> {
    private static final long serialVersionUID = 1L;

    public ValueMap() {}

    public ValueMap withValue(String key, String val) {
        super.put(key, val);
        return this;
    }

... Add withXYZ to suit...

}

Потім ви можете реалізувати його так:

ValueMap map = new ValueMap()
      .withValue("key 1", "value 1")
      .withValue("key 2", "value 2")
      .withValue("key 3", "value 3")

0

Це я завжди хотів, особливо під час встановлення тестових приладів. Нарешті, я вирішив написати простий вільний конструктор, який міг би створити будь-яку реалізацію карти - https://gist.github.com/samshu/b471f5a2925fa9d9b718795d8bbdfe42#file-mapbuilder-java

    /**
     * @param mapClass Any {@link Map} implementation type. e.g., HashMap.class
     */
    public static <K, V> MapBuilder<K, V> builder(@SuppressWarnings("rawtypes") Class<? extends Map> mapClass)
            throws InstantiationException,
            IllegalAccessException {
        return new MapBuilder<K, V>(mapClass);
    }

    public MapBuilder<K, V> put(K key, V value) {
        map.put(key, value);
        return this;
    }

    public Map<K, V> build() {
        return map;
    }

0

Ось один я написав

import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import java.util.function.Supplier;

public class MapBuilder<K, V> {

    private final Map<K, V> map;

    /**
     * Create a HashMap builder
     */
    public MapBuilder() {
        map = new HashMap<>();
    }

    /**
     * Create a HashMap builder
     * @param initialCapacity
     */
    public MapBuilder(int initialCapacity) {
        map = new HashMap<>(initialCapacity);
    }

    /**
     * Create a Map builder
     * @param mapFactory
     */
    public MapBuilder(Supplier<Map<K, V>> mapFactory) {
        map = mapFactory.get();
    }

    public MapBuilder<K, V> put(K key, V value) {
        map.put(key, value);
        return this;
    }

    public Map<K, V> build() {
        return map;
    }

    /**
     * Returns an unmodifiable Map. Strictly speaking, the Map is not immutable because any code with a reference to
     * the builder could mutate it.
     *
     * @return
     */
    public Map<K, V> buildUnmodifiable() {
        return Collections.unmodifiableMap(map);
    }
}

Ви використовуєте його так:

Map<String, Object> map = new MapBuilder<String, Object>(LinkedHashMap::new)
    .put("event_type", newEvent.getType())
    .put("app_package_name", newEvent.getPackageName())
    .put("activity", newEvent.getActivity())
    .build();

0

Використання java 8:

Це підхід Java-9 Map.ofEntries(Map.entry(k1,v1), Map.entry(k2,v2), ...)

public class MapUtil {
    import static java.util.stream.Collectors.toMap;

    import java.util.AbstractMap.SimpleEntry;
    import java.util.Map;
    import java.util.Map.Entry;
    import java.util.stream.Stream;

    private MapUtil() {}

    @SafeVarargs
    public static Map<String, Object> ofEntries(SimpleEntry<String, Object>... values) {
        return Stream.of(values).collect(toMap(Entry::getKey, Entry::getValue));
    }

    public static SimpleEntry<String, Object> entry(String key, Object value) {
        return new SimpleEntry<String, Object>(key, value);
    }
}

Як використовувати:

import static your.package.name.MapUtil.*;

import java.util.Map;

Map<String, Object> map = ofEntries(
        entry("id", 1),
        entry("description", "xyz"),
        entry("value", 1.05),
        entry("enable", true)
    );


0

Підкреслення-Java може створювати хешмап.

Map<String, Object> value = U.objectBuilder()
        .add("firstName", "John")
        .add("lastName", "Smith")
        .add("age", 25)
        .add("address", U.arrayBuilder()
            .add(U.objectBuilder()
                .add("streetAddress", "21 2nd Street")
                .add("city", "New York")
                .add("state", "NY")
                .add("postalCode", "10021")))
        .add("phoneNumber", U.arrayBuilder()
            .add(U.objectBuilder()
                .add("type", "home")
                .add("number", "212 555-1234"))
            .add(U.objectBuilder()
                .add("type", "fax")
                .add("number", "646 555-4567")))
        .build();
    // {firstName=John, lastName=Smith, age=25, address=[{streetAddress=21 2nd Street,
    // city=New York, state=NY, postalCode=10021}], phoneNumber=[{type=home, number=212 555-1234},
    // {type=fax, number=646 555-4567}]}
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.