I am pretty sure they are not the same. However, I am bogged down by the common notion that "Rust does not support" higher-kinded types (HKT), but instead offers parametric polymorphism. I tried to get my head around that and understand the difference between these, but got just more and more entangled.
To my understanding, there are higher-kinded types in Rust, at least the basics. Using the "*"-notation, a HKT does have a kind of e.g. * -> *
.
For example, Maybe
is of kind * -> *
and could be implemented like this in Haskell.
data Maybe a = Just a | Nothing
Here,
Maybe
is a type constructor and needs to be applied to a concrete type to become a concrete type of kind "*".Just a
andNothing
are data constructors.
In textbooks about Haskell, this is often used as an example for a higher-kinded type. However, in Rust it can simply be implemented as an enum, which after all is a sum type:
enum Maybe<T> {
Just(T),
Nothing,
}
Where is the difference? To my understanding this is a perfectly fine example of a higher-kinded type.
- If in Haskell this is used as a textbook example of HKTs, why is it
said that Rust doesn't have HKT? Doesn't the
Maybe
enum qualify as a HKT? - Should it rather be said that Rust doesn't fully support HKT?
- What's the fundamental difference between HKT and parametric polymorphism?
This confusion continues when looking at functions, I can write a parametric
function that takes a Maybe
, and to my understanding a HKT as a function
argument.
fn do_something<T>(input: Maybe<T>) {
// implementation
}
again, in Haskell that would be something like
do_something :: Maybe a -> ()
do_something :: Maybe a -> ()
do_something _ = ()
which leads to the fourth question.
- Where exactly does the support for higher-kinded types end? Whats the minimal example to make Rust's type system fail to express HKT?
Related Questions:
I went through a lot of questions related to the topic (including links they have to blogposts, etc.) but I could not find an answer to my main questions (1 and 2).
- In Haskell, are "higher-kinded types" *really* types? Or do they merely denote collections of *concrete* types and nothing more?
- Generic struct over a generic type without type parameter
- Higher Kinded Types in Scala
- What types of problems helps "higher-kinded polymorphism" solve better?
- Abstract Data Types vs. Parametric Polymorphism in Haskell
Update
Thank you for the many good answers which are all very detailed and helped a lot. I decided to accept Andreas Rossberg's answer since his explanation helped me the most to get on the right track. Especially the part about terminology.
I was really locked in the cycle of thinking that everything of kind * -> * ... -> *
is higher-kinded. The explanation that stressed the difference between * -> * -> *
and (* -> *) -> *
was crucial for me.
* -> *
or higher. – ScentT
has to be a concrete type and cannot be higher kinded itself? Aka. something likestruct Baz<T<V>> {}
will not work? – Fatten->
s (much as a higher-order function is a function which takes parameters with types containing->
s). By that definitionMaybe
is not a higher-kinded type, since its one argument is a regular type of kind*
. – Tamayo[]
is another regular type constructor (its parameter is also kind*
).Monad
is an example of a higher kinded type class, because itsm
parameter has a kind of* -> *
(Monad :: (* -> *) -> Constraint
). – Tamayostruct Baz<C<_>> { C<A> a; C<B> b; }
-- note that the parameter C is not a type, but something that maps types to types. Scala has HKT (if you enable them), and Haskell has them as well. – Bidentate