为了理解两个发电机

时间:2015-07-26 21:59:58

标签: scala for-comprehension

我想在垂直列表中的GUI应用程序中显示任意数量的字符串。

考虑以下示例:

val names = Seq[String]("Boris", "Anna", "Markus", "Simone")

for(a <- 1 to names.size + 1; name <- names) yield new ListItem(
  name,
  x = 20
  y = 128 * a
  size = 128
)

当然,这将创建一个“跨产品”并打印:

Boris Anna Markus Simone
Boris Anna Markus Simone
Boris Anna Markus Simone
Boris Anna Markus Simone

虽然想要的结果是

Boris
Anna
Markus
Simone

但是如何将索引绑定到实际变量呢?我的意思是,如果我做一个像这样的计数器:

for(a <- 0 to 1; b <- 0 to 1)
yield(a,b)

我会

Vector((0,0), (0,1), (1,0), (1,1))

但是,如果我想,我可以(for a <- 0 to 1; b <- 0 to a) yield(a,b)并获得

Vector((0,0), (1,0), (1,1))

我不知道如何用我的例子来做到这一点。

1 个答案:

答案 0 :(得分:3)

您需要zipWithIndex,这将创建一个新的Seq,其中包含原始Seq中的每个元素,并在Seq内与其进行了索引。< / p>

scala> names.zipWithIndex
res2: Seq[(String, Int)] = List((Boris,0), (Anna,1), (Markus,2), (Simone,3))

然后您可以使用:

val names = Seq[String]("Boris", "Anna", "Markus", "Simone")
class ListItem(name: String, x: Int, y: Int, size: Int)

scala> for ((name, index) <- names.zipWithIndex) 
           yield new ListItem(name, 20, 128 * index, 128)
res0: Seq[ListItem] = List(ListItem@5956b37d, ListItem@4b22015d, ListItem@2587a734, ListItem@6cf25a2b)

也可以写成map

names.zipWithIndex.map { case (name, index) =>
    new ListItem(name, 20, 128 * index, 128)
}