模式匹配与Scala专业

时间:2016-01-23 02:29:25

标签: scala pattern-matching bytecode boxing specialized-annotation

我有一个必须使用Double和Float的类。由于性能要求,我将@specialized注释用于(Double, Float)。我需要调用两个第三方功能。 ffunc(x: Float)接受Floatdfunc(y: Double)接受Double。在某些时候,我必须拨打ffuncdfunc。我为此目的使用scala模式匹配。我的代码如下:

class BoxingTest[@specialized(Double, Float) T] {
  def foo(p: T): Unit = {
    p match {
      case dp: Double => dfunc(dp)
      case df: Float => ffunc(df)
    }
  }
}

但是,scala编译器为专用版本提供了非优化的字节码。当我查看专用类的字节码时,我看到了非优化的匹配代码,它将我的专用类型转换为对象。还有额外的装箱/拆箱如下:

41: invokestatic  #21                 // Method scala/runtime/BoxesRunTime.boxToDouble:(D)Ljava/lang/Double;
44: invokestatic  #39                 // Method scala/runtime/BoxesRunTime.unboxToDouble:(Ljava/lang/Object;)D

你能否建议用优化的代码替换匹配代码,避免装箱/拆箱?

1 个答案:

答案 0 :(得分:1)

之前出现了这个问题。我认为最好的办法是覆盖专门的方法:

scala> class X[@specialized(Int, Double) A] { def f(a: A): String = ??? }
defined class X

scala> trait T { def f$mcI$sp(i: Int): String = "my int" }
defined trait T

scala> val x = new X[Int] with T
x: X[Int] with T = $anon$1@6137cf6e

scala> x.f(42)
res0: String = my int

那可能是在SO。