Guava предоставляет нам отличные методы factory для типов Java, например Maps.newHashMap()
.
Но существуют ли также конструкторы для java-карт?
HashMap<String,Integer> m = Maps.BuildHashMap.
put("a",1).
put("b",2).
build();
Guava предоставляет нам отличные методы factory для типов Java, например Maps.newHashMap()
.
Но существуют ли также конструкторы для java-карт?
HashMap<String,Integer> m = Maps.BuildHashMap.
put("a",1).
put("b",2).
build();
Поскольку интерфейс Java 9 Map
содержит:
Map.of(k1,v1, k2,v2,..)
Map.ofEntries(Map.entry(k1,v1), Map.entry(k2,v2),..)
.Ограничения этих заводских методов заключаются в том, что они:
null
качестве ключей и/или значений (если вам нужно сохранить нули, посмотрите на другие ответы) Если нам нужны изменяемые карты, мы можем использовать конструктор копирования выбранной изменяемой карты и передавать туда результат Map.of(..)
как
Map<Integer, String> map = new HashMap<>( Map.of(1,"a", 2,"b", 3,"c") );
Для 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());
Не совсем построитель, но с использованием инициализатора:
Map<String, String> map = new HashMap<String, String>() {{
put("a", "1");
put("b", "2");
}};
Это похоже на принятый ответ, но немного чище, на мой взгляд:
ImmutableMap.of("key1", val1, "key2", val2, "key3", val3);
Существует несколько вариантов описанного выше метода, и они отлично подходят для создания статических неизменных неизменяемых карт.
Простой конструктор карт тривиально писать:
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());
}
}
}
Вот очень простой...
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.
Вы можете использовать:
HashMap<String,Integer> m = Maps.newHashMap(
ImmutableMap.of("a",1,"b",2)
);
Это не так классно и читаемо, но делает работу.
HashMap
является изменяемым; нет необходимости в строителе.
Map<String, Integer> map = Maps.newHashMap();
map.put("a", 1);
map.put("b", 2);
У меня было подобное требование некоторое время назад. Это не имеет ничего общего с 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")
Это то, что я всегда хотел, особенно при настройке тестовых приборов. Наконец, я решил написать простой свободный строитель, который мог бы построить любую реализацию Карты - 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;
}
Вот один я написал
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();