使用Iterator将元组映射到元组

时间:2015-04-21 14:10:35

标签: scala iterator

为什么以下代码不起作用,如何使用Iterator克服它?

def f(str : String) : (String, String) = {
  str.splitAt(1)
}
var with_id : Iterator[(String, Int)] = List(("test", 1), ("list", 2), ("nothing", 3), ("else", 4)).iterator

println(with_id.mkString(" "))

val result = with_id map { (s : String, i : Int) => (f(s), i) }

println(result.mkString(" "))

预期输出为:

(("t", "est"), 1) (("l", "ist"), 2) ...

错误:

Error:(28, 54) type mismatch;
found   : (String, Int) => ((String, String), Int)
required: ((String, Int)) => ?
val result = with_id map { (s : String, i : Int) => (f(s), i) }
                                                 ^

2 个答案:

答案 0 :(得分:2)

问题是(s : String, i : Int) => (f(s), i)Function2(即一个带2个参数的函数):

scala> (s : String, i : Int) => (f(s), i)
res3: (String, Int) => ((String, String), Int) = <function2>

.map期望Function1(以元组为参数)。

您可以使用

定义Function1
scala> val g = (t: (String, Int)) => (f(t._1), t._2)
g: ((String, Int)) => ((String, String), Int) = <function1>

scala> val result = with_id map g
result: Iterator[((String, String), Int)] = non-empty iterator

但是(至少对我来说)使用更加惯用的模式匹配匿名函数似乎要好得多(注意添加的case):

scala> val result = with_id map { case (s : String, i : Int) => (f(s), i) }
result: Iterator[((String, String), Int)] = non-empty iterator

答案 1 :(得分:1)

with_id.map期望((String, Int) => ?)函数作为输入。也就是说,一个函数采用Tuple作为输入,而不是两个参数。

你可以像这样使用它:

with_id map{ case (s,i) => (f(s), i)}  //match the input tuple to s and i