Scala Итератор с картой и для

Дано:

val list = List("one","two","three")     
val it = list.toIterator

Я могу запустить:

list map ("_" +) -> List(_one, _two, _three)
for (i <- list) yield("_" + i) -> List(_one, _two, _three)

Если я запустил то же самое на итераторе, я получаю:

it map ("_" + ) -> Iterator[java.lang.String] = empty iterator
for (i <- it) yield("_" + i) -> Iterator[java.lang.String] = empty iterator

Не следует ли мне вернуть еще один (не пустой) Iterator [String] после запуска карты/на нем?

Ответ 1

scala> def ints(n: Int): Stream[Int] = n #:: ints(n + 1)
ints: (n: Int)Stream[Int]

scala> val list = List("one","two","three")
list: List[java.lang.String] = List(one, two, three)

scala> val it = list.toIterator
it: Iterator[java.lang.String] = non-empty iterator

scala> it map ("_" + )
res24: Iterator[java.lang.String] = non-empty iterator

scala> it map ("_" + )
res25: Iterator[java.lang.String] = non-empty iterator

scala> for (i <- it) yield("_" + i)
res26: Iterator[java.lang.String] = non-empty iterator

Возможно, вы использовали свой итератор?

scala> res26.foreach{println}
_one
_two
_three

scala> res26
res28: Iterator[java.lang.String] = empty iterator

Так как итераторы имеют состояние с сохранением состояния и не сбрасываются, как только вы его использовали, он пуст и больше не может быть использован.

Вместо этого вы можете использовать представления:

scala> val v = list.view
v: java.lang.Object with scala.collection.SeqView[java.lang.String,List[java.lang.String]] = SeqView(one, two, three)

scala> v map ("_" + )
res29: scala.collection.SeqView[java.lang.String,Seq[_]] = SeqViewM(...)

scala> for (i <- v) yield("_" + i)
res30: scala.collection.SeqView[java.lang.String,Seq[_]] = SeqViewM(...)

scala> res29.foreach{println}
_one
_two
_three

scala> res29
res32: scala.collection.SeqView[java.lang.String,Seq[_]] = SeqViewM(...)

scala> res29.foreach{println}
_one
_two
_three

Ответ 2

См. Итераторы.

Существует важное различие между методом foreach на итераторах и тем же методом в проходящих коллекциях: при вызове к итератору foreach оставит итератор в конце, когда это будет сделано. Поэтому вызов next снова на том же итераторе завершится неудачно с помощью NoSuchElementException. Напротив, при вызове в коллекции foreach количество элементов в коллекции остается неизменным (если только переданная функция не добавляет к удалению элементов, но это обескураживает, потому что это может привести к неожиданным результатам).

...

Как вы можете видеть, после вызова it.map итератор it продвинулся до конца.