Kotlin is a powerful and modern programming language that makes working with generics easier than Java. One of the most important concepts when dealing with generics in Kotlin is covariance. If you’ve ever seen the out
keyword in Kotlin and wondered what it does, this blog is for you. Let’s break down Kotlin covariance in a simple, easy-to-understand way.
What is Kotlin Covariance?
Covariance refers to preserving the subtyping relation between generic classes. In Kotlin, you can declare a class to be covariant on a specific type parameter by using the out
keyword before the type parameter’s name.
A covariant class is a generic class (we’ll use Producer as an example) for which the following holds: Producer<A> is a subtype of Producer<B> if A is a subtype of B. We say that the subtyping is preserved. For example, Producer<Cat> is a subtype of Producer<Animal> because Cat is a subtype of Animal.
Here’s an example of the Producer
interface using the out
keyword:
interface Producer<out T> {
fun produce(): T
}
Flexible Function Argument and Return Value Passing
Covariance in Kotlin allows you to pass values of a class as function arguments and return values, even when the type arguments don’t exactly match the function definition. This means that you can use a more specific type as a substitute for a more generic type.
Suppose we have a hierarchy of classes involving Animal
, where Cat
is a subclass of Animal
. We also have a generic interface called Producer
, which represents a producer of objects of type T
. We’ll make the Producer
interface covariant by using the out
keyword on the type parameter.
interface Producer<out T> {
fun produce(): T
}
Now, let’s define a class AnimalProducer
that implements the Producer
interface for the Animal
type:
class AnimalProducer : Producer<Animal> {
override fun produce(): Animal {
return Animal()
}
}
Since Animal
is a subtype of Animal
, we can use AnimalProducer
wherever a Producer<Animal>
is expected.
Now, let’s define another class CatProducer
that implements the Producer
interface for the Cat
type:
class CatProducer : Producer<Cat> {
override fun produce(): Cat {
return Cat()
}
}
Since Cat
is a subtype of Animal
, we can also use CatProducer
wherever a Producer<Animal>
is expected. This is possible because we declared the Producer
interface as covariant.
Now, let’s see how covariance allows us to pass these producers as function arguments and return values:
fun feedAnimal(producer: Producer<Animal>) {
val animal = producer.produce()
animal.feed()
}
fun main() {
val animalProducer = AnimalProducer()
val catProducer = CatProducer()
feedAnimal(animalProducer) // Passes an AnimalProducer, which is a Producer<Animal>
feedAnimal(catProducer) // Passes a CatProducer, which is also a Producer<Animal>
}
In the feedAnimal
function, we expect a Producer<Animal>
as an argument. With covariance, we can pass both AnimalProducer
and CatProducer
instances because Producer<Cat>
is a subtype of Producer<Animal>
due to the covariance declaration.
This demonstrates how covariance allows you to treat more specific types (Producer<Cat>
) as if they were more generic types (Producer<Animal>
) when it comes to function arguments and return values.
BTW, How covariance guarantees type safety?
Suppose we have a class hierarchy involving Animal
, where Cat
is a subclass of Animal
. We also have a Herd
class that represents a group of animals.
open class Animal {
fun feed() { /* feeding logic */ }
}
class Herd<T : Animal> { // The type parameter isn't declared as covariant
val size: Int get() = /* calculate the size of the herd */
operator fun get(i: Int): T { /* get the animal at index i */ }
}
fun feedAll(animals: Herd<Animal>) {
for (i in 0 until animals.size) {
animals[i].feed()
}
}
Now, suppose you have a function called takeCareOfCats
, which takes a Herd<Cat>
as a parameter and performs some operations specific to cats.
class Cat : Animal() {
fun cleanLitter() { /* clean litter logic */ }
}
fun takeCareOfCats(cats: Herd<Cat>) {
for (i in 0 until cats.size) {
cats[i].cleanLitter()
// feedAll(cats) // This line would cause a type-mismatch error, Error: inferred type is Herd<Cat>, but Herd<Animal> was expected
}
}
In this case, if you try to pass the cats
herd to the feedAll
function, you’ll get a type-mismatch error during compilation. This happens because you didn’t use any variance modifier on the type parameter T
in the Herd
class, making the Herd<Cat>
incompatible with Herd<Animal>
. Although you could use an explicit cast to overcome this issue, it is not a recommended approach.
To make it work correctly, you can make the Herd
class covariant by using the out
keyword on the type parameter:
class Herd<out T : Animal> { // The T parameter is now covariant.
// ...
}
fun takeCareOfCats(cats: Herd<Cat>) {
for (i in 0 until cats.size) {
cats[i].cleanLitter()
}
feedAll(cats) // Now this line works because of covariance, You don't need a cast.
}
By marking the type parameter as covariant, you ensure that the subtyping relation is preserved, and T
can only be used in “out” positions. This guarantees type safety and allows you to pass a Herd<Cat>
where a Herd<Animal>
is expected.
Usage of covariance
Covariance in Kotlin allows you to make a class covariant on a type parameter, but it also imposes certain constraints to ensure type safety. The type parameter can only be used in “out” positions, which means it can produce values of that type but not consume them.
You can’t make any class covariant: it would be unsafe. Making the class covariant on a certain type parameter constrains the possible uses of this type parameter in the class. To guarantee type safety, it can be used only in so-called out positions, meaning the class can produce values of type T but not consume them. Uses of a type parameter in declarations of class members can be divided into “in” and “out” positions.
Let’s consider a class that declares a type parameter T and contains a function that uses T. We say that if T is used as the return type of a function, it’s in the out position. In this case, the function produces values of type T. If T is used as the type of a function parameter, it’s in the in position. Such a function consumes values of type T.

The function parameter type is called in position, and the function return type is called out position
The out keyword on a type parameter of the class requires that all methods using T have T only in “out” positions and not in “in” positions. This keyword constrains possible use of T, which guarantees safety of the corresponding subtype relation.
Let’s understand this with some examples. Consider the Herd
class, which is declared as Herd<out T : Animal>
. The type parameter T
is used only in the return value of the get
method. This is an “out” position, and it is safe to declare the class as covariant. For instance, Herd<Cat>
is considered a subtype of Herd<Animal>
because Cat
is a subtype of Animal
.
class Herd<out T : Animal> {
val size: Int = ...
operator fun get(i: Int): T { ... } // Uses T as the return type
}
Similarly, the List<T>
interface in Kotlin is covariant because it only defines a get
method that returns an element of type T
. Since there are no methods that store values of type T
, it is safe to declare the class as covariant.
interface List<out T> : Collection<T> {
operator fun get(index: Int): T // Read-only interface that defines only methods that return T (so T is in the “out” position)
// ...
}
You can also use the type parameter T
as a type argument in another type. For example, the subList
method in the List
interface returns a List<T>
, and T
is used in the “out” position.
interface List<out T> : Collection<T> {
fun subList(fromIndex: Int, toIndex: Int): List<T> // Here T is in the “out” position as well.
// ...
}
However, you cannot declare MutableList<T>
as covariant on its type parameter because it contains methods that both consume and produce values of type T
. Therefore, T
appears in both “in” and “out” positions, and making it covariant would be unsafe.
interface MutableList<T> : List<T>, MutableCollection<T> { //MutableList can’t be declared as covariant on T …
override fun add(element: T): Boolean // … because T is used in the “in” position.
}
The compiler enforces this restriction. It would report an error if the class was declared as covariant: Type parameter T is declared as ‘out’ but occurs in ‘in’ position.
Why and When to Use Kotlin Covariance?
Covariance helps to preserve subtyping relationships. Without covariance, the Kotlin compiler would not allow a subtype to be assigned to a supertype in a generic context.
Use out
when:
- Your generic type is a producer of values.
- You only need to return values of the generic type.
- You want to enable subtype assignments safely.
Conclusion
Covariance in Kotlin simplifies working with generics while maintaining type safety. By using the out
keyword, you can allow subtype assignments in a controlled manner. Understanding Kotlin Covariance helps you write more flexible and safe generic code, making your Kotlin programs more robust and maintainable.
Start using covariance in your Kotlin projects today and experience the power of type-safe, flexible generics..!