【发布时间】:2014-08-19 23:25:36
【问题描述】:
让我们定义一个PartialFunction[String, String] 和一个PartialFunction[Any, String]
现在,给定orElse的定义
def orElse[A1 <: A, B1 >: B](that: PartialFunction[A1, B1]): PartialFunction[A1, B1]
我希望不能组合这两者,因为
A → StringA1 → Any
因此绑定的A1 <: A(即Any <: String)不成立。
没想到,我可以组合它们并获得一个在整个String 域上定义的PartialFunction[String, String]。这是一个例子:
val a: PartialFunction[String, String] = { case "someString" => "some other string" }
// a: PartialFunction[String,String] = <function1>
val b: PartialFunction[Any, String] = { case _ => "default" }
// b: PartialFunction[Any,String] = <function1>
val c = a orElse b
// c: PartialFunction[String,String] = <function1>
c("someString")
// res4: String = some other string
c("foo")
// res5: String = default
c(42)
// error: type mismatch;
// found : Int(42)
// required: String
此外,如果我明确提供orElse 类型参数
a orElse[Any, String] b
// error: type arguments [Any,String] do not conform to method orElse's type parameter bounds [A1 <: String,B1 >: String]
编译器终于有点意思了。
我是否缺少任何类型系统巫术导致b 成为orElse 的有效参数?也就是说A1怎么会被推断为String?
如果编译器从b 推断出A1,那么它一定是Any,那么导致String 的推断链从哪里开始?
更新
在玩过 REPL 之后,我注意到 orElse 在类型不匹配时返回一个交集类型 A with A1。示例:
val a: PartialFunction[String, String] = { case "someString" => "some other string" }
// a: PartialFunction[String,String] = <function1>
val b: PartialFunction[Int, Int] = { case 42 => 32 }
// b: PartialFunction[Int,Int] = <function1>
a orElse b
// res0: PartialFunction[String with Int, Any] = <function1>
由于(String with Int) <:< String 这有效,即使生成的函数实际上无法使用。我还怀疑String with Any 被统一为Any,因为
import reflect.runtime.universe._
// import reflect.runtime.universe._
typeOf[String] <:< typeOf[String with Any]
// res1: Boolean = true
typeOf[String with Any] <:< typeOf[String]
// res2: Boolean = true
这就是为什么将String 和Any 混合成String 的原因。
话虽如此,引擎盖下发生了什么?不匹配的类型是在什么逻辑下统一的?
更新 2
我已将问题简化为更一般的形式:
class Foo[-A] {
def foo[B <: A](f: Foo[B]): Foo[B] = f
}
val a = new Foo[Any]
val b = new Foo[String]
a.foo(b) // Foo[String] Ok, String <:< Any
b.foo(a) // Foo[String] Shouldn't compile! Any <:!< String
b.foo[Any](a) // error: type arguments [Any] do not conform to method foo's type parameter bounds [A <: String]
【问题讨论】:
-
很确定编译器只是在推断它可以找到的最严格的返回类型。
-
而不是向后推断其他类型?
-
Nitpick: "String with Int" 不是联合类型——根据定义——作为一个类型需要 "String" 和 "Int" 独立地实现 "String with Int" 的接口,这并非如此。 "String" 不是 <: with int>
-
感谢@Utaal,它确实是一个交集而不是一个联合。修好了!
标签: scala types partialfunction