理解无形的“单态”示例

Understanding `Monomorphism` Example of Shapeless

Shapeless Features Overview 显示以下示例:

import poly._

// choose is a function from Sets to Options with no type specific cases
object choose extends (Set ~> Option) {
  def apply[T](s : Set[T]) = s.headOption
}

scala> choose(Set(1, 2, 3))
res0: Option[Int] = Some(1)

scala> choose(Set('a', 'b', 'c'))
res1: Option[Char] = Some(a)

但是,由于缺乏使用 Shapeless 的经验,我不了解它与以下内容之间的区别:

scala> def f[T](set: Set[T]): Option[T] = set.headOption
f: [T](set: Set[T])Option[T]

scala> f( Set(1,2,3) )
res0: Option[Int] = Some(1)

scala> f( Set('a', 'b', 'c') )
res1: Option[Char] = Some(a)

这里的重要区别是,choose 是一个可以作为值传递的函数。你不能从 f 中得到一个(合理的)值,因为 Scala 不支持多态函数值:

scala> val fun = f _
fun: Set[Nothing] => Option[Nothing] = <function1>

如您所见,Scala 将元素类型固定为 Nothing 使函数对非空集无用:

scala> fun(Set(1))
<console>:10: error: type mismatch;
 found   : Int(1)
 required: Nothing
              fun(Set(1))
                      ^

这与您对 Shapeless 方法的预期一样有效。