Skip to content

Contact sales

By filling out this form and clicking submit, you acknowledge our privacy policy.

Covariance and Contravariance for C#

Get clarity on covariance and contravariance in C#, which were introduced with version 4.0.

Dec 12, 2019 • 4 Minute Read

Introduction

With the introduction of C# version 4.0, two new features were added to the already rich set of tools the language provides to solve different problems. The features are covariance and contravariance. This guide will clarify those terms, and show you in real code how they enrich the feature set and allow developers to provide solutions to problems that previously required a workaround.

The co and contra-variances refer to how you treat an object. Immutable collection classes like lists are covariant when you are only allowed to push elements in if they are of type T or its subtype. Covariance comes into picture when you are working with mutable collections.

Variance as a term means a property of operators that act on specific types and the ordering of operators. The co prefix means "together with", and informs you that the operator preserves the ordering of types when compared to its opernands. The contra prefix means "against" something, and tells you that the operator reverses the ordering of types when compared to its operands.

Variance: Either Co or Contra

The following problem revolves around Servers and Network devices. Let's say you have a list of devices that have names and states. When making a list, include the devices you intend to process and ensure the devices have unique names and states.

Here is a demonstration for covariance:

      List<Devices> switches = GetSwitches();
    

With C# 3.0, this will not compile as the compiler cannot figure out what you can or cannot do with the switches collection after you have it. Nothing would prevent you from adding a server to your collection, which means the following is not guaranteed to not happen.

      switches.Add(new Server("Domain Controller","up"));
    

The compiler refuses to compile the code. This demonstrates contravariance.

In the following example, the compiler will allow the code, even though the method returns List< Devices >. This is because all switches are devices.

      List<Switch> switches = GetAccess();
switches.Add(new Switch("DC-CORE-1","up"));
    

We can change types.

      List<Devices> switches = GetAccess();
switches.Add(new Switch("DC-CORE-1","up"));
    

However this is also ambiguous because you could mean the following as well:

      List<Switch> switches = GetAccess();
Switch dccore = switches[0];
    

Version 4.0 introduced two new keywords: in and out. They refer to the kind of arguments that can be placed in certain positions. The in marks input-position, the out marks output-position.

This allows us to code our original intentions.

      IEnumerable<Devices> devices = GetSwitches();
    

The List has both the out and in directions, meaning it is neither co- nor contra-variant. Take the following:

      interface IOnlyDevices<in T>{
	void Add(T value);
}
    

The simple interface makes it possible to do the following:

      IOnlyDevices<Switch> switches = GetAccess();
switches.Add(new Switch("DC-CORE-1","up"));
    

So now the last statement is safe and unambiguous.

Conclusion

Contravariance allows you to utilize a less derived type than originally specified, and covariance lets you use a more derived type. In a sense, the reason they were brought to the C# language is so you can extend arrays, delegate types and generic types with polymorphistic features. Choose co- or contra-variance based on the type of data you are working with.

Dániel Szabó

Dániel Szabó

Written content author.

More about this author