I am researching about existential types in Scala 2.12.x
. For that I'm testing the following code:
trait Parent
class ChildA extends Parent
class ChildB extends Parent
def whatIsInside(opt: Option[_ <: Parent]): String = {
opt match {
case _: Option[_ <: ChildA] => "ChildA"
case _: Option[_ <: ChildB] => "ChildB"
case _ => throw new IllegalArgumentException("unknown type")
}
}
whatIsInside(Some(new ChildA))
I don't expect this to work at runtime because of type erasure, but nevertheless this does not even compile. I am getting the following error:
[error] ExistentialTypes.scala:12:24: not found: type _$2
[error] case _: Option[_ <: ChildA] => "ChildA"
[error] ^
[error] ExistentialTypes.scala:13:24: not found: type _$3
[error] case _: Option[_ <: ChildB] => "ChildB"
[error] ^
Can someone explain these errors?
(Not a full answer, but a few notes and links; Maybe it can serve as a starting point for someone else)
In 2.12.13, the compiler seems to be able to prove that
F[_ <: X]
andF[X]
are the same type ifX
occurs in covariant position:This compiles (with warnings, but it compiles):
This does not compile:
So, it seems that it must have something to do with the bounds inference for the synthetic
_$2
type-variable.Also, this compiles:
so, unless I'm misinterpreting the spec:
, the
pt
would beOption[_ <: Parent]
, thena1,...,an
would be the single synthetic type_$2
, and the constraints_$2 >: Nothing <: ChildA
should make the typeOption[_$2]
conform toOption[_ <: Parent]
. So, it seems that it should work, but doesn't.Bonus
If you just wanted to make it work, then just skip all those wildcards, they aren't needed: