Преобразовать java.util.HashMap в scala.collection.immutable.Map в java

Я использую некоторую библиотеку Scala из моего кода Java. И у меня проблема с коллекциями. Мне нужно передать scala.collection.immutable.Map в качестве параметра метода. Я могу преобразовать или построить immutable.Map из моего Java-кода, но я не знаю, как это сделать. Предложения?

Ответы

Ответ 1

Полностью можно использовать JavaConverters в Java-коде: есть только несколько дополнительных обручей, которые можно перепрыгнуть через:

import java.util.HashMap;
import scala.Predef;
import scala.Tuple2;
import scala.collection.JavaConverters;
import scala.collection.immutable.Map;

public class ToScalaExample {
  public static <A, B> Map<A, B> toScalaMap(HashMap<A, B> m) {
    return JavaConverters.mapAsScalaMapConverter(m).asScala().toMap(
      Predef.<Tuple2<A, B>>conforms()
    );
  }

  public static HashMap<String, String> example() {
    HashMap<String, String> m = new HashMap<String, String>();
    m.put("a", "A");
    m.put("b", "B");
    m.put("c", "C");
    return m;
  }
}

Мы можем показать, что это работает из Scala REPL:

scala> val jm: java.util.HashMap[String, String] = ToScalaExample.example
jm: java.util.HashMap[String,String] = {b=B, c=C, a=A}

scala> val sm: Map[String, String] = ToScalaExample.toScalaMap(jm)
sm: Map[String,String] = Map(b -> B, c -> C, a -> A)

Но, конечно, вы могли бы так же легко вызвать эти методы из кода Java.

Ответ 2

Можете ли вы предоставить дополнительный вызов API, который принимает/предоставляет java.util.Map, преобразованный с помощью JavaConverters?

class Example {
   import scala.collection.JavaConverters._
   def fromMap(m:Map[...]) = ...

   // generics etc. elided
   def fromJava(m:java.util.Map) = {
      fromMap(m.asScala.toMap)
   }
}

Возможно, вы захотите извлечь конверсию и предоставить декоратор (особенно, поскольку я отмечаю, что вы работаете с библиотекой Scala). Обратите внимание на комментарий dhg re. неизменность.

Ответ 3

Это работало для меня с java 1.8 и scala 2.12:

public static <K, V> scala.collection.immutable.Map<K, V> toScalaImmutableMap(java.util.Map<K, V> jmap) {
    List<Tuple2<K, V>> tuples = jmap.entrySet()
      .stream()
      .map(e -> Tuple2.apply(e.getKey(), e.getValue()))
      .collect(Collectors.toList());

    Seq<Tuple2<K, V>> scalaSeq = JavaConverters.asScalaBuffer(tuples).toSeq();

    return (Map<K, V>) Map$.MODULE$.apply(scalaSeq);
}

Ответ 4

Мое решение для Java 1.7 и Scala 2.11:

@SuppressWarnings("unchecked")
private static <K, V> scala.collection.immutable.Map<K, V> toScalaImmutableMap(java.util.Map<K, V> javaMap) {
    final java.util.List<scala.Tuple2<K, V>> list = new java.util.ArrayList<>(javaMap.size());
    for (final java.util.Map.Entry<K, V> entry : javaMap.entrySet()) {
        list.add(scala.Tuple2.apply(entry.getKey(), entry.getValue()));
    }
    final scala.collection.Seq<Tuple2<K, V>> seq = scala.collection.JavaConverters.asScalaBufferConverter(list).asScala().toSeq();
    return (scala.collection.immutable.Map<K, V>) scala.collection.immutable.Map$.MODULE$.apply(seq);
}