for-comprehension中的scala异常与类型注释

时间:2015-01-05 23:01:21

标签: scala for-comprehension scala-2.11

我试图理解在处理空值并在for-comprehension中输入注释时看起来很奇怪的行为。

举个例子:

def f(): String = null

for {
  a <- Option("hello")
  b = f()
} yield (a, b)

导致预期:

//> res0: Option[(String, String)] = Some((hello,null)) 

但是,如果我在类型b

中添加类型注释
def f(): String = null

for {
  a <- Option("hello")
  b: String = f()
} yield (a, b)

然后我得到一个运行时异常:

//> scala.MatchError: (hello,null) (of class scala.Tuple2)

为什么会这样?在第一个示例中,b隐式地不是String?第二个示例中的显式类型注释会发生什么变化?

(注意,示例在Scala 2.11.4中运行)

1 个答案:

答案 0 :(得分:7)

null不是任何事情的实例:

scala> (null: String) match { case _: String => }
scala.MatchError: null
  ... 33 elided

scala> val s: String = null
s: String = null

scala> s.isInstanceOf[String]
res1: Boolean = false

http://www.scala-lang.org/files/archive/spec/2.11/08-pattern-matching.html#type-patterns

类型模式指定非空。

显示翻译的一个技巧是评论显示:

scala> for {
     |   a <- Option("hello")
     |   b: String = f()
     | } yield (a, b) // show
object $read extends scala.AnyRef {
  def <init>() = {
    super.<init>;
    ()
  };
  object $iw extends scala.AnyRef {
    def <init>() = {
      super.<init>;
      ()
    };
    import $line4.$read.$iw.$iw.f;
    object $iw extends scala.AnyRef {
      def <init>() = {
        super.<init>;
        ()
      };
      val res1 = Option("hello").map(((a) => {
        val b: String = f;
        scala.Tuple2(a, b)
      })).map(((x$1) => x$1: @scala.unchecked match {
        case scala.Tuple2((a @ _), (b @ (_: String))) => scala.Tuple2(a, b)
      }))
    }
  }
}
scala.MatchError: (hello,null) (of class scala.Tuple2)
  at $anonfun$2.apply(<console>:10)
  at $anonfun$2.apply(<console>:10)
  at scala.Option.map(Option.scala:145)
  ... 39 elided