了解Case Class的'unapply'

问题描述 投票:3回答:1

在下面的代码中,我试图理解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字节码指令?

scala
1个回答
1
投票

如果使用-Xprint:typer-print进行编译,您将看到生成的代码。这是用apply编译的unapplyscalac -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

这里的模式匹配变成了简单的类型检查。

© www.soinside.com 2019 - 2024. All rights reserved.