In the following simplified sample code:
case class One[A](a: A) // An identity functor
case class Twice[F[_], A](a: F[A], b: F[A]) // A functor transformer
type Twice1[F[_]] = ({type L[α] = Twice[F, α]}) // We'll use Twice1[F]#L when we'd like to write Twice[F]
trait Applicative[F[_]] // Members omitted
val applicativeOne: Applicative[One] = null // Implementation omitted
def applicativeTwice[F[_]](implicit inner: Applicative[F]): Applicative[({type L[α] = Twice[F, α]})#L] = null
I can call applicativeTwice on applicativeOne, and type inference works, as soon as I try to call it on applicativeTwice(applicativeOne), inference fails:
val aOK = applicativeTwice(applicativeOne)
val bOK = applicativeTwice[Twice1[One]#L](applicativeTwice(applicativeOne))
val cFAILS = applicativeTwice(applicativeTwice(applicativeOne))
The errors in scala 2.10.0 are
- type mismatch;
found : tools.Two.Applicative[[α]tools.Two.Twice[tools.Two.One,α]]
required: tools.Two.Applicative[F]
- no type parameters for method applicativeTwice:
(implicit inner: tools.Two.Applicative[F])tools.Two.Applicative[[α]tools.Two.Twice[F,α]]
exist so that it can be applied to arguments
(tools.Two.Applicative[[α]tools.Two.Twice[tools.Two.One,α]])
--- because ---
argument expression's type is not compatible with formal parameter type;
found : tools.Two.Applicative[[α]tools.Two.Twice[tools.Two.One,α]]
required: tools.Two.Applicative[?F]
Why wouldn't "?F" match with anything (of the right kind) ? Ultimately I'd like applicativeTwice to be an implicit function, but I'd have to get the type inference working first. I have seen similar questions, and the answers pointed to limitations in the type inference algorithms. But this case seems pretty limitative, and must be quite an annoyance in monad transformers, so I suspect I'm missing some trick to work around this.
You've hit a common annoyance: SI-2712. For clarity, I'm going to minimize your code a bit:
This demonstrates the same type error as yours:
First a bit of syntax and terminology you probably already know:
Int
) has kind_
. It's monomorphic.Base
, on the other hand, is parameterized. we can't use it as the type of a value without providing the type it contains, so we say has kind_[_]
. It's rank-1 polymorphic: a type constructor that takes a type.Recursive
goes further still: it has two parameters,F[_]
andA
. The number of type parameters don't matter here, but their kinds do.F[_]
is rank-1 polymorphic, soRecursive
is rank-2 polymorphic: it's a type constructor that takes a type constructor.Scala in general doesn't have trouble with higher-kinded types. This is one of several key features that distinguishes its type system from, say, Java's. But it does have trouble with partial application of type parameters when dealing with higher-kinded types.
Here's the problem:
Recursive[F[_], A]
has two type parameters. In your sample code, you did the "type lambda" trick to partially apply the first parameter, something like:This convinces the compiler that you're providing something of the correct kind (
_[_]
) to theRecursive
constructor. If Scala had curried type parameter lists, I'd definitely have used that here:Alas, it does not (see SI-4719). So, to the best of my knowledge, the most common way of dealing with this problem is the "unapply trick," due to Miles Sabin. Here is a greatly simplified version of what appears in scalaz:
In somewhat hand-wavey terms, this
Unapply
construct is like a "first-class type lambda." We define a trait representing the assertion that some typeFA
can be decomposed into a type constructorF[_]
and a typeA
. Then in its companion object, we can define implicits to provide specific decompositions for types of various kinds. I've only defined here the specific one that we need to makeRecursive
fit, but you could write others.With this extra bit of plumbing, we can now do what we need:
Ta-da! Now type inference works, and this compiles. As an exercise, I'd suggest you create an additional class:
... which isn't really different from
Recursive
in any meaningful way, of course, but will again break type inference. Then define the additional plumbing needed to fix it. Good luck!Edit
You asked for a less simplified version, something aware of type-classes. Some modification is required, but hopefully you can see the similarity. First, here's our upgraded
Unapply
:Again, this is completely ripped off from scalaz. Now some sample code using it: