Преобразование коллекции Java в коллекцию Scala

88

Связанный с вопросом о переполнении стека. Scala-эквивалент нового HashSet (Collection) , как мне преобразовать коллекцию Java ( java.util.Listскажем) в коллекцию Scala List?

На самом деле я пытаюсь преобразовать вызов Java API в Spring SimpleJdbcTemplate , который возвращает a java.util.List<T>, в неизменяемый Scala HashSet. Так например:

val l: java.util.List[String] = javaApi.query( ... )
val s: HashSet[String] = //make a set from l

Кажется, это работает. Критика приветствуется!

import scala.collection.immutable.Set
import scala.collection.jcl.Buffer 
val s: scala.collection.Set[String] =
                      Set(Buffer(javaApi.query( ... ) ) : _ *)
Oxbow_lakes
источник

Ответы:

17

Ваше последнее предложение работает, но вы также можете избежать использования jcl.Buffer:

Set(javaApi.query(...).toArray: _*)

Обратите внимание, что scala.collection.immutable.Setпо умолчанию он доступен благодаря Predef.scala.

Хорхе Ортис
источник
9
Это предложение не работает там, где я хочу сохранить информацию о типе
oxbow_lakes,
127

Для справки в будущем: в Scala 2.8 это можно было бы сделать так:

import scala.collection.JavaConversions._
val list = new java.util.ArrayList[String]()
list.add("test")
val set = list.toSet

setэто scala.collection.immutable.Set[String]после этого.

Также см. Ответ Бена Джеймса для более явного способа (с использованием JavaConverters), который, похоже, рекомендуется сейчас.

робинст
источник
Похоже, теперь JavaConversionsесть некоторые последствия, делающие toSetзвонок ненужным.
Rajish
@Rajish Я думаю , что лучше , если преобразование явно (см stackoverflow.com/questions/8301947/... )
krookedking
вы правы, лучше использовать import scala.collection.JavaConverters._
Абхишек Сенгупта
62

Если вы хотите быть более явным, чем JavaConversions, продемонстрированные в ответе robinst , вы можете использовать JavaConverters:

import scala.collection.JavaConverters._
val l = new java.util.ArrayList[java.lang.String]
val s = l.asScala.toSet
Бен Джеймс
источник
25

JavaConversions ( ответ робинста ) и JavaConverters ( ответ Бена Джеймса ) устарели в Scala 2.10 .

Вместо JavaConversions используйте:

import scala.collection.convert.wrapAll._

как предложил aleksandr_hramcov .

Вместо JavaConverters используйте:

import scala.collection.convert.decorateAll._

Для обоих также есть возможность импортировать преобразования / конвертеры только в Java или Scala соответственно, например:

import scala.collection.convert.wrapAsScala._

Обновление: приведенное выше утверждение, что JavaConversions и JavaConverters устарели, кажется неправильным. В Scala 2.10 были некоторые устаревшие свойства, из-за которых при их импорте появлялись предупреждения об устаревании. Таким образом, альтернативный импорт здесь, кажется, только псевдонимы. Хотя я все же предпочитаю их, так как ИМХО названия более подходящие.

стемплер
источник
Почему вы говорите, что JavaConversions и JavaConverters устарели? В текущей документации 2.11.8 оба не помечены как устаревшие, и в них также нет ссылки на decorateAll или wrapAll, указывающую, что это новое и предпочтительное использование.
Joost den Boer
@JoostdenBoer Кажется , там были некоторые устаревшие декораторы там в Scala 2.10, хотя они , кажется, были снова удалены Scala 2.11.
Stempler
14

Вы также можете изучить эту отличную библиотеку: scalaj-collection, которая имеет двустороннее преобразование между коллекциями Java и Scala. В вашем случае, чтобы преобразовать java.util.List в Scala List, вы можете сделать это:

val list = new java.util.ArrayList[java.lang.String]
list.add("A")
list.add("B")
list.asScala
Сурья Сураварапу
источник
1
Эта библиотека - величайшая вещь на свете. Действительно действительно хорошо работает.
Майкл Нил
7
Возвращаясь к этому через пару лет, можно сказать, что JavaConverters в Scala - это то, что нужно.
Сурья Сураварапу
5

При запуске Scala 2.13пакет scala.jdk.CollectionConvertersзаменяет пакеты scala.collection.JavaConverters/JavaConversions._:

import scala.jdk.CollectionConverters._

// val javaList: java.util.List[String] = java.util.Arrays.asList("one","two","three")
javaList.asScala
// collection.mutable.Buffer[String] = Buffer("one", "two", "three")
javaList.asScala.toSet
// collection.immutable.Set[String] = Set("one", "two", "three")
Ксавье Гихот
источник
3
val array = java.util.Arrays.asList("one","two","three").toArray

val list = array.toList.map(_.asInstanceOf[String])
Jamesqiu
источник
это отличный ответ и решил мою проблему. Большинство методов, упомянутых в Интернете, просто преобразуют (например) java.util.List [java.lang.Long] в scala.collection.immutable.List [java.lang.Long]. Итак, этот метод помог мне преобразовать его в scala.collection.immutable.List [scala.Long]
ajkl
2

Вы можете добавить информацию о типе в вызов toArray, чтобы параметризовать Set:

 val s = Set(javaApi.query(....).toArray(new Array[String](0)) : _*)

Это может быть предпочтительным , так как пакет коллекции происходит через основные переделки для Scala 2.8 и пакет scala.collection.jcl это уходит

Аарон
источник
1

Вы можете преобразовать коллекцию Java в массив, а затем создать из этого список Scala:

val array = java.util.Arrays.asList("one","two","three").toArray
val list = List.fromArray(array)
Фабиан Штег
источник
2
Это не очень хорошо, потому что мой java.util.List возвращается из Java API в виде параметризованного списка (поэтому мой вызов API дает java.util.List <String>) - я пытаюсь повернуть это в неизменяемый HashSet scala
oxbow_lakes,
1

Еще один простой способ решить эту проблему:

import collection.convert.wrapAll._
aleksandr_hramcov
источник