了解Case Class的'unapply'

时间:2016-05-20 02:44:23

标签: scala

在下面的代码中,我试图了解Foo#unapply : Option[Int]在模式匹配方面的用法。

$ls
Test.scala
$cat Test.scala 
object Test {

  case class Foo(x: Int)

  def f(foo: Foo): Unit = foo match {
    case Foo(_) => ()
  }

}

它编译:

$scalac Test.scala 
$ls
Test$.class Test$Foo$.class Test$Foo.class  Test.class  Test.scala

我在每个javap -c -v X.class | grep unapply文件上运行了*class。但是,我没有找到任何结果。

请解释Foo#unapply如何在模式匹配中使用。另外,我在哪里可以找到相应的JVM字节码指令?

1 个答案:

答案 0 :(得分:1)

如果您使用-Xprint:typer-print进行编译,则会看到生成的代码。这是applyunapply的已清理版本,使用scalac -Xprint:typer Test.scala编译:

case def apply(x: Int): com.yuval.Test.Foo = new Foo(x);
case def unapply(x$0: com.yuval.Test.Foo): Option[Int] = if (x$0.==(null))
    scala.this.None
  else
    Some.apply[Int](x$0.x);

f仍然相同:

def f(foo: com.yuval.Test.Foo): Unit = foo match {
  case (x: Int)com.yuval.Test.Foo(_) => ()
}

这是来自Test$.class的Java反编译代码:

public void f(Test.Foo foo) {
    Test.Foo localFoo = foo;
    if (localFoo != null) {
        BoxedUnit localBoxedUnit = BoxedUnit.UNIT;
    } else {
        throw new MatchError(localFoo);
    }
}

这是使用java -c -p Test$.class生成的字节码:

public void f(com.yuval.Test$Foo);
  Code:
   0: aload_1
   1: astore_2
   2: aload_2
   3: ifnull        11
   6: getstatic     #20    // Field scala/runtime/BoxedUnit.UNIT:Lscala/runtime/BoxedUnit;
   9: astore_3
  10: return
  11: new           #22    // class scala/MatchError
  14: dup
  15: aload_2
  16: invokespecial #25    // Method scala/MatchError."<init>":(Ljava/lang/Object;)V
  19: athrow

此处的模式匹配变为简单的类型检查。