我最近看过这样的代码:
val maybeInt = catching(classOf[NFE]) opt arg.toInt
Run Code Online (Sandbox Code Playgroud)
这是什么opt?一个选项?为什么不使用getOrElse来提取值?在上面的代码中,maybeInt如果抛出NumberFormatException ,则为None?
Rex*_*err 36
catching看起来它是某种方法调用,不是吗?它是,但它实际上返回一个类的实例Catch; 它没有直接参与论证.这个类有两个方法,对于处理异常特别有用(还有几个用于捕获多个异常).首先是
def opt [U >: T] (body: ? U) : Option[U]
Run Code Online (Sandbox Code Playgroud)
这里使用的是 - 你给它一些可能抛出异常的东西,Some(result)如果一切正常,它会返回,None如果抓住了目标异常:
scala> type NFE = NumberFormatException
defined type alias NFE
scala> import scala.util.control.Exception._
import scala.util.control.Exception._
scala> catching(classOf[NFE]).opt( "fish".toInt )
res0: Option[Int] = None
scala> catching(classOf[NFE]).opt( "42".toInt )
res1: Option[Int] = Some(42)
Run Code Online (Sandbox Code Playgroud)
然后,您可以使用map或filter或getOrElse用于处理选项的其他任何内容来处理此问题.
另一个有用的方法是either返回Left(exception)抛出异常的实例,如果Right(result)不是则返回:
scala> catching(classOf[NFE]).either( "fish".toInt )
res2: Either[Throwable,Int] = Left(java.lang.NumberFormatException: For input string: "fish")
scala> catching(classOf[NFE]).either( "42".toInt )
res3: Either[Throwable,Int] = Right(42)
Run Code Online (Sandbox Code Playgroud)
然后,您可以使用fold或映射到选项或您喜欢的其他任何操作.
请注意,您可以定义单个捕获器并多次使用它(因此您不必每次都创建捕获器对象,例如,解析整数):
scala> val catcher = catching(classOf[NFE])
catcher: util.control.Exception.Catch[Nothing] = Catch(java.lang.NumberFormatException)
scala> catcher.opt("42".toInt)
res4: Option[Int] = Some(42)
scala> catcher.opt("fish".toInt)
res5: Option[Int] = None
Run Code Online (Sandbox Code Playgroud)
编辑:丹尼尔在评论中指出,这仍然是一个暂时的Catch[Option]; 给定方法签名,没有一种简单的方法可以让它捕获异常并生成选项而不创建任何额外的对象.这让我想起为什么我会编写自己的方法来做到这一点:
def optNFE[T](t: => T) = try { Some(t) } catch {case nfe: NFE => None}
optNFE( "fish".toInt ) // gives None
optNFE( "42".toInt ) // gives Some(42)
Run Code Online (Sandbox Code Playgroud)