В чем разница между JavaConverters и JavaConversions в Scala?

В scala.collection есть два очень похожих объекта JavaConversions и JavaConverters.

  • В чем разница между этими двумя объектами?
  • Почему они оба существуют?
  • Когда я хочу использовать одно и другое?

Ответ 1

РЕДАКТИРОВАТЬ: Java Conversions получили @deprecated в Scala 2.13.0. Вместо этого используйте scala.jdk.CollectionConverters.

JavaConversions предоставляют серию неявных методов, которые преобразуют между коллекцией Java и ближайшей соответствующей коллекцией Scala, и наоборот. Это делается путем создания оболочек, которые реализуют либо интерфейс Scala и перенаправляют вызовы в базовую коллекцию Java, либо в интерфейс Java, перенаправляя вызовы в базовую коллекцию Scala.

JavaConverters использует JavaConverters -my-library для "добавления" метода asScala к коллекциям Java и метода asJava к коллекциям Scala, которые возвращают соответствующие оболочки, описанные выше. Он новее (начиная с версии 2.8.1), чем JavaConversions (начиная с версии 2.8), и делает преобразование между Scala и коллекцией Java явным. Вопреки тому, что Дэвид пишет в своем ответе, я бы порекомендовал вам использовать привычку использовать JavaConverters как вы с гораздо меньшей вероятностью будете писать код, который выполняет много неявных преобразований, поскольку вы можете контролировать единственное место, где это будет случается: где вы пишете .asScala или .asJava.

Вот методы преобразования, которые предоставляют JavaConverters:

Pimped Type                            | Conversion Method   | Returned Type
=================================================================================================
scala.collection.Iterator              | asJava              | java.util.Iterator
scala.collection.Iterator              | asJavaEnumeration   | java.util.Enumeration
scala.collection.Iterable              | asJava              | java.lang.Iterable
scala.collection.Iterable              | asJavaCollection    | java.util.Collection
scala.collection.mutable.Buffer        | asJava              | java.util.List
scala.collection.mutable.Seq           | asJava              | java.util.List
scala.collection.Seq                   | asJava              | java.util.List
scala.collection.mutable.Set           | asJava              | java.util.Set
scala.collection.Set                   | asJava              | java.util.Set
scala.collection.mutable.Map           | asJava              | java.util.Map
scala.collection.Map                   | asJava              | java.util.Map
scala.collection.mutable.Map           | asJavaDictionary    | java.util.Dictionary
scala.collection.mutable.ConcurrentMap | asJavaConcurrentMap | java.util.concurrent.ConcurrentMap
—————————————————————————————————————————————————————————————————————————————————————————————————
java.util.Iterator                     | asScala             | scala.collection.Iterator
java.util.Enumeration                  | asScala             | scala.collection.Iterator
java.lang.Iterable                     | asScala             | scala.collection.Iterable
java.util.Collection                   | asScala             | scala.collection.Iterable
java.util.List                         | asScala             | scala.collection.mutable.Buffer
java.util.Set                          | asScala             | scala.collection.mutable.Set
java.util.Map                          | asScala             | scala.collection.mutable.Map
java.util.concurrent.ConcurrentMap     | asScala             | scala.collection.mutable.ConcurrentMap
java.util.Dictionary                   | asScala             | scala.collection.mutable.Map
java.util.Properties                   | asScala             | scala.collection.mutable.Map[String, String]

Однако для использования преобразований непосредственно из Java лучше вызывать методы непосредственно из JavaConversions; например:

List<String> javaList = new ArrayList<String>(Arrays.asList("a", "b", "c"));
System.out.println(javaList); // [a, b, c]
Buffer<String> scalaBuffer = JavaConversions.asScalaBuffer(javaList);
System.out.println(scalaBuffer); // Buffer(a, b, c)
List<String> javaListAgain = JavaConversions.bufferAsJavaList(scalaBuffer);
System.out.println(javaList == javaListAgain); // true

Ответ 2

Для тех, кто приступает к этому вопросу, поскольку Scala 2.12.x, JavaConversions теперь устарел, а JavaConverters - предпочтительный метод.

Ответ 3

Как объясняется в API, JavaConversions - это набор неявных преобразований, который преобразует коллекции java в связанную коллекцию scala.

Вы можете использовать его с import collection.JavaConversions._. При необходимости компилятор автоматически преобразует java-коллекцию в правильный тип scala.

JavaConverters - это набор декораторов, который помогает преобразовывать коллекции java или scala в коллекции scala или java, используя asScala или asJava, которые будут неявно добавлены в коллекцию, которую вы хотите преобразовать. Чтобы использовать эти конвертеры, вам необходимо импортировать:

import collection.JavaConverters._

Вы должны предпочесть JavaConversions, поскольку он обычно проще в использовании (нет необходимости использовать asScala или asJava).

Ответ 4

В Scala 2.13 JavaConverters устарели в пользу scala.jdk.CollectionConverters:

... новый пакет scala.jdk с объектами CollectionConverters (классические коллекции Java, аналогичные collection.JavaConverters в 2.12), StreamConverters, FunctionConverters и OptionConverters...