从这个 sbt console 输出中可以看出:
scala> :type implicit class Harden[S <% mutable.Set[_]](val set: S)
[S]AnyRef {
val set: S
private[this] val set: S
implicit private[this] val evidence$1: S => scala.collection.mutable.Set[_]
def <init>(set: S)(implicit evidence$1: S => scala.collection.mutable.Set[_]): Harden[S]
}
实际上,Harden
的构造函数在幕后被解析为:
def <init>(set: S)(implicit evidence$1: S => scala.collection.mutable.Set[_]): Harden[S]
如在 value classes limitations 中所述:
必须只有一个主构造函数,该构造函数恰好有一个公共的 val 参数,其类型不是一个值类。
这意味着 Harden
违反了此限制。
但是您可以实现类似的功能。尝试将类上定义的视图界定转换为方法中的隐式证明。
类似于下面这样:
scala> implicit class Harden[S](val set: S) extends AnyVal {
| def cast[T](implicit ev: S => scala.collection.mutable.Set[_]) = set.map(_.asInstanceOf[T])
| }
defined class Harden
这将编译通过:
scala> Set(1,2,3).cast[Any]
res17: scala.collection.mutable.Set[Any] = Set(1, 2, 3)
这将会失败,正如预期的一样:
scala> List(1,2,3).cast[Any]
<console>:24: error: No implicit view available from List[Int] => scala.collection.mutable.Set[_].
List(1,2,3).cast[Any]
^
<%
)将在2.12中被弃用。只是让你知道。 - Ryoichiro Oka