scala - 如何使用高阶函数展平选项列表?

标签 scala scala-option

使用 Scala 2.7.7:

如果我有一个选项列表,我可以使用 for-comprehension 将它们展平:

val listOfOptions = List(None, Some("hi"), None)
listOfOptions: List[Option[java.lang.String]] = List(None, Some(hi), None)

scala> for (opt <- listOfOptions; string <- opt) yield string
res0: List[java.lang.String] = List(hi)

我不喜欢这种风格,宁愿使用 HOF。这种尝试过于冗长,无法接受:
scala> listOfOptions.flatMap(opt => if (opt.isDefined) Some(opt.get) else None)
res1: List[java.lang.String] = List(hi)

直觉上,我本希望以下内容有效,但事实并非如此:
scala> List.flatten(listOfOptions)
<console>:6: error: type mismatch;
 found   : List[Option[java.lang.String]]
 required: List[List[?]]
       List.flatten(listOfOptions)

即使以下内容似乎也应该起作用,但不起作用:
scala> listOfOptions.flatMap(_: Option[String])
<console>:6: error: type mismatch;
 found   : Option[String]
 required: (Option[java.lang.String]) => Iterable[?]
       listOfOptions.flatMap(_: Option[String])
                          ^

我能想到的最好的是:
scala> listOfOptions.flatMap(_.toList)         
res2: List[java.lang.String] = List(hi)

...但我宁愿不必将选项转换为列表。那看起来很笨重。

有什么建议吗?

最佳答案

在 Scala 2.8 中,flatten 将起作用:


Welcome to Scala version 2.8.0.RC2 (Java HotSpot(TM) 64-Bit Server VM, Java 1.6.0_20).
Type in expressions to have them evaluated.
Type :help for more information.

scala> val listOfOptions = List(None, Some("hi"), None)
listOfOptions: List[Option[java.lang.String]] = List(None, Some(hi), None)

scala> listOfOptions flatten
res0: List[java.lang.String] = List(hi)

但是,这在 2.7.7 中不起作用:

Welcome to Scala version 2.7.7.final (Java HotSpot(TM) 64-Bit Server VM, Java 1.6.0_20).

scala> val listOfOptions = List(None, Some("hi"), None)
listOfOptions: List[Option[java.lang.String]] = List(None, Some(hi), None)

scala> listOfOptions.flatten
:6: error: no implicit argument matching parameter type (Option[java.lang.String]) => Iterable[Nothing] was found.
       listOfOptions.flatten

集合库已经过重新设计,并在 2.8 中得到了很大改进,所以也许您可能想尝试使用最新的 Scala 2.8 RC,看看它是否使您更容易使用。

如果你真的不想使用 toList 方法,我猜你也可以这样写:

scala> listOfOptions.flatMap(o => o)
res: List[java.lang.String] = List(hi)

也许也不是什么美事,但至少这在 2.7.7 中有效。

关于scala - 如何使用高阶函数展平选项列表?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2895069/

相关文章:

java - Scala+Android+Roboguice : ClassNotFoundException in loader

java - Scala 参数传递给 Java (Play Framework)

scala - 为什么 Scala 选项的 foreach 比 get 更好?

java - 可选化 Java getter

scala - 可以/应该在 Scala 中添加/创建从 T 到 Option[T] 的隐式转换吗?

database - 如何使用 actor 进行数据库访问和 DDD?

java - 如何将 zip 文件的内容分配给 Spark 中的每个任务?

scala - 添加reactivemongo后出现错误 "play-iteratees_2.10 not found"

scala - 如何将 scala 中的序列转换/包装为 Option[Seq],以便如果列表为空,则 Option 为 None

斯卡拉 : Nested getOrElse