使用partial-function-parameter方法的基类型的Scala值类编译失败

use*_*360 5 scala partialfunction higher-order-functions value-class

比如说,我定义了一个值类,如下所示

package object p {
  class ValueClass[T](val o: Option[T]) extends AnyVal {
    def foo: Option[T] =
      o collect {
        case t => t
      }
  }
}
Run Code Online (Sandbox Code Playgroud)

编译失败并显示以下消息:

overriding method applyOrElse in trait PartialFunction of type [A1 <: T, B1 >: T](x: A1, default: A1 => B1)B1;
  method applyOrElse has incompatible type
       o collect {
                 ^
Run Code Online (Sandbox Code Playgroud)

错误消息对我来说似乎没那么有意义,就好像我替换collectmap(需要函数而不是部分函数)或类没有扩展AnyVal,代码片段将编译.

有谁可以解释背后的原因或链接到提交的问题?

som*_*ytt 6

这是一个很好的.它的工作原理是2.11.0-M8,至少.

显然,

https://issues.scala-lang.org/browse/SI-8011

与完全匹配

https://issues.scala-lang.org/browse/SI-8018

在2.10.x,

scala> trait X[A] { def pf: PartialFunction[A,A] = { case a => a } }
defined trait X

scala> class X[A](val x: A) extends AnyVal { def pf: PartialFunction[A,A] = { case a => a } }
<console>:7: error: overriding method applyOrElse in trait PartialFunction of type [A1 <: A, B1 >: A](x: A1, default: A1 => B1)B1;
 method applyOrElse has incompatible type
       class X[A](val x: A) extends AnyVal { def pf: PartialFunction[A,A] = { case a => a } }
                                                                            ^
Run Code Online (Sandbox Code Playgroud)

也许这个问题在2.10.3允许之后没有出现.

对于病态的好奇或空闲,以前有过:

https://issues.scala-lang.org/browse/SI-6482

https://issues.scala-lang.org/browse/SI-7022 (重复)

https://issues.scala-lang.org/browse/SI-6187

好的,官方验证它实际上是一个好的.这是后端口被移植的提交:

commit ff9f60f420c090b6716c927ab0359b082f2299de
Author: Paul Phillips <paulp@improving.org>
Date:   Sat Oct 6 10:20:45 2012 -0700

    Fix for SI-6482, lost bounds in extension methods.

    That was a good one. How to create a new method with type
    parameters from multiple sources, herein.
Run Code Online (Sandbox Code Playgroud)

更新显示2.11.0-M8做了一些事情:

apm@mara:~/goof$ scalam
Welcome to Scala version 2.11.0-M8 (OpenJDK 64-Bit Server VM, Java 1.7.0_25).
Type in expressions to have them evaluated.
Type :help for more information.

scala> class X[A](val x: A) extends AnyVal { def pf: PartialFunction[A,A] = { case a => a } }
defined class X

scala> :pa -raw
// Entering paste mode (ctrl-D to finish)

package object p {
  class ValueClass[T](val o: Option[T]) extends AnyVal {
    def foo: Option[T] =
      o collect {
        case t => t
      }
  }
}

// Exiting paste mode, now interpreting.


scala> new p.ValueClass(Some(7)).foo
res0: Option[Int] = Some(7)
Run Code Online (Sandbox Code Playgroud)