2

假设我有一个列表(或地图中的值),并且我想对每个项目执行操作。但不幸的是,无论出于何种原因,这个值列表都可能包含空值。

scala> val players = List("Messi", null, "Xavi", "Iniesta", null)
players: List[java.lang.String] = List(Messi, null, Xavi, Iniesta, null)

为了避免被 NPE 炸毁,我需要执行以下操作:

scala> players.filterNot(_ == null ).map(_.toUpperCase)
res84: List[java.lang.String] = List(MESSI, XAVI, INIESTA)

有没有更好的方法来做到这一点?

理想情况下是这样的:

players.safeMap(_.toUpperCase)

scala-language 邮件列表中,Simon 提出了这个:

players.filter ( null !=).map(_.toUpperCase )

这是我最初拍摄的较短版本,并且在没有专用方法的情况下尽可能短。

更好的是,Stefan 和 Kevin 提出了 withFilter 方法,该方法将返回一个惰性代理,因此可以合并两个操作。

players.withFilter ( null !=).map(_.toUpperCase )
4

3 回答 3

3

如果您无法避免nulls (例如,如果您从 Java 代码中获取列表),另一种选择是使用collect而不是map

scala> players.collect { case player if player != null => player.toUpperCase }
res0: List[java.lang.String] = List(MESSI, XAVI, INIESTA)
于 2012-04-26T21:08:08.260 回答
2

我会这样做:

players flatMap Option map (_.toUpperCase)

但这比collect. filter+map总是更好地完成collect.

于 2012-04-27T00:25:21.507 回答
1

您可以转换为以下列表Option[String]

scala> val optionPlayers = players.map(Option(_))
optionPlayers: List[Option[java.lang.String]] = List(Some(Messi), None, Some(Xavi), Some(Iniesta), None)

Option是普遍首选的null,它在如何安全处理数据方面为您提供了很大的灵活性。以下是获得所需结果的简单方法:

scala> optionPlayers.collect { case Some(s) => s.toUpperCase }
res0: List[java.lang.String] = List(MESSI, XAVI, INIESTA)

scala> optionPlayers.flatMap(_.map(_.toUpperCase))
res1: List[java.lang.String] = List(MESSI, XAVI, INIESTA)

scala> optionPlayers.flatten.map(_.toUpperCase)
res2: List[java.lang.String] = List(MESSI, XAVI, INIESTA)

Option您可以在其他 StackOverflow 问题中或通过搜索网络找到更多信息。

或者,您始终可以将safeMap所需的方法定义为隐式 on List

implicit def enhanceList[T](list: List[T]) = new {
  def safeMap[R](f: T => R) = list.filterNot(_ == null).map(f)
}

所以你可以这样做:

scala> players.safeMap(_.toUpperCase)
res4: List[java.lang.String] = List(MESSI, XAVI, INIESTA)

虽然如果你定义一个隐式,你可能想要使用CanBuildFrom像基本集合一样的样式来使它不仅仅适用于List. 您可以在其他地方找到更多相关信息。

于 2012-04-26T21:02:30.193 回答