<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>Forem: Amita Shukla</title>
    <description>The latest articles on Forem by Amita Shukla (@amita__shukla).</description>
    <link>https://forem.com/amita__shukla</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F29000%2F42689d4d-f6a5-4742-8342-076a7e857e72.jpg</url>
      <title>Forem: Amita Shukla</title>
      <link>https://forem.com/amita__shukla</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://forem.com/feed/amita__shukla"/>
    <language>en</language>
    <item>
      <title>Understanding Types, Subtypes and Type Variance</title>
      <dc:creator>Amita Shukla</dc:creator>
      <pubDate>Sun, 02 May 2021 13:17:20 +0000</pubDate>
      <link>https://forem.com/amita__shukla/understanding-types-subtypes-and-type-variance-1gl6</link>
      <guid>https://forem.com/amita__shukla/understanding-types-subtypes-and-type-variance-1gl6</guid>
      <description>&lt;p&gt;This is a cross-post from my &lt;a href="https://amitashukla.in/blog"&gt;blog&lt;/a&gt;. You can find the original &lt;a href="https://amitashukla.in/blog/understanding-type-subtypes-variance/"&gt;here&lt;/a&gt;.&lt;/p&gt;

&lt;p&gt;Most of us when we just start with programming, get confused with concepts of types, subtypes, inheritance, etc. But just as with the process of learning any language this all starts to come naturally to us, we grow up to casually use subtyping wherever we want in a program. In fact, it becomes almost unimaginable to code without these.  &lt;/p&gt;

&lt;p&gt;Subtyping is nothing but writing one type in terms of another. Only when I started exploring other languages I realized that there are deeper theories to it, and maybe I do not understand subtypes to the extent I thought I did. I am writing this post in order to pass on this programming existential crisis to my fellow readers (but also attempting to develop an understanding along the way). We shall take subtyping a level further and understand how the assignment and substitution behaviour of our types affects our code behaviour (and subsequently ours).&lt;/p&gt;

&lt;p&gt;Let's begin with an example. Let's say we have a class &lt;code&gt;Animal&lt;/code&gt;. It can &lt;code&gt;speak&lt;/code&gt; and &lt;code&gt;eat&lt;/code&gt;.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="k"&gt;class&lt;/span&gt; &lt;span class="nc"&gt;Animal&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;name&lt;/span&gt;&lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;String&lt;/span&gt;&lt;span class="o"&gt;){&lt;/span&gt;
  &lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;speak&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="s"&gt;"animal speaking"&lt;/span&gt;
  &lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;eat&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="s"&gt;"animal eats"&lt;/span&gt;
  &lt;span class="k"&gt;override&lt;/span&gt; &lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;toString&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="s"&gt;"animal: "&lt;/span&gt; &lt;span class="o"&gt;+&lt;/span&gt; &lt;span class="n"&gt;name&lt;/span&gt;
&lt;span class="o"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;While &lt;code&gt;Animal&lt;/code&gt; has general properties, it can also be of two types: &lt;code&gt;Cat&lt;/code&gt; and &lt;code&gt;Dog&lt;/code&gt;. &lt;code&gt;Cat&lt;/code&gt; and &lt;code&gt;Dog&lt;/code&gt; &lt;code&gt;speak&lt;/code&gt;, &lt;code&gt;eat&lt;/code&gt; in their own way. Also, a dog has its own special function as well, &lt;code&gt;dogonly&lt;/code&gt;. Similarly, a cat is special in its own way, say &lt;code&gt;catonly&lt;/code&gt;.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="k"&gt;class&lt;/span&gt; &lt;span class="nc"&gt;Cat&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="k"&gt;override&lt;/span&gt; &lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;name&lt;/span&gt;&lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;String&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt; &lt;span class="k"&gt;extends&lt;/span&gt; &lt;span class="nc"&gt;Animal&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;name&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt; &lt;span class="o"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;override&lt;/span&gt; &lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;speak&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="s"&gt;"meow"&lt;/span&gt;
  &lt;span class="k"&gt;override&lt;/span&gt; &lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;eat&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="s"&gt;"fish"&lt;/span&gt;
  &lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;catonly&lt;/span&gt; &lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;String&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="s"&gt;"cat only"&lt;/span&gt;
&lt;span class="o"&gt;}&lt;/span&gt;

&lt;span class="k"&gt;class&lt;/span&gt; &lt;span class="nc"&gt;Dog&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="k"&gt;override&lt;/span&gt; &lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;name&lt;/span&gt;&lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;String&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt; &lt;span class="k"&gt;extends&lt;/span&gt; &lt;span class="nc"&gt;Animal&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;name&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt; &lt;span class="o"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;override&lt;/span&gt; &lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;speak&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="s"&gt;"woof"&lt;/span&gt;
  &lt;span class="k"&gt;override&lt;/span&gt; &lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;eat&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="s"&gt;"bone"&lt;/span&gt;
  &lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;dogonly&lt;/span&gt; &lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;String&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="s"&gt;"dog only"&lt;/span&gt;
&lt;span class="o"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;(&lt;em&gt;I do not have experience with pets, so pardon me if I did not implement Dog's and Cat's functions correctly...&lt;/em&gt;)&lt;/p&gt;

&lt;p&gt;With the above examples in mind, let's go over some concepts formally.&lt;/p&gt;

&lt;h2&gt;
  
  
  Type
&lt;/h2&gt;

&lt;p&gt;A type or a data type represents the type of data that tells the compiler/interpreter as to how the programmer intends to use that data. So, a type can be &lt;code&gt;Integer&lt;/code&gt;, &lt;code&gt;String&lt;/code&gt;, &lt;code&gt;Boolean&lt;/code&gt;. We can create our own types as well, such as &lt;code&gt;Animal&lt;/code&gt;, &lt;code&gt;Cat&lt;/code&gt; and &lt;code&gt;Dog&lt;/code&gt; as above. A type, the way it is defined, indicates what values it can take and what operations can be done on it.&lt;/p&gt;

&lt;h2&gt;
  
  
  Subtyping
&lt;/h2&gt;

&lt;p&gt;Subtyping talks about the relationship between different types. We have a &lt;strong&gt;subtype&lt;/strong&gt; and we have a &lt;strong&gt;supertype&lt;/strong&gt;.&lt;/p&gt;

&lt;p&gt;We can say that a type &lt;code&gt;B&lt;/code&gt; is a subtype of &lt;code&gt;A&lt;/code&gt; if:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;we can substitute &lt;code&gt;B&lt;/code&gt; subtype in place of &lt;code&gt;A&lt;/code&gt; supertype. &lt;/li&gt;
&lt;li&gt;we can perform all the operations on &lt;code&gt;B&lt;/code&gt; that can be performed on &lt;code&gt;A&lt;/code&gt;. &lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;This is often written as &lt;code&gt;A :&amp;gt; B&lt;/code&gt;, meaning that an instance of type &lt;code&gt;B&lt;/code&gt; can be safely used wherever type &lt;code&gt;A&lt;/code&gt; is expected. &lt;/p&gt;

&lt;p&gt;Let's be very clear on this as this will be the basis of our article: &lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;If &lt;code&gt;Animal :&amp;gt; Cat&lt;/code&gt;, then we can use &lt;code&gt;Cat&lt;/code&gt; wherever we expect an &lt;code&gt;Animal&lt;/code&gt;. &lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;So, if we have a list that expects objects of &lt;code&gt;Animal&lt;/code&gt; type:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;animal1&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;Animal&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="s"&gt;"animal1"&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt;
&lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;animal2&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;Animal&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="s"&gt;"animal2"&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt;
&lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;animals&lt;/span&gt; &lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;List&lt;/span&gt;&lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="kt"&gt;Animal&lt;/span&gt;&lt;span class="o"&gt;]&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="nc"&gt;List&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;animal1&lt;/span&gt;&lt;span class="o"&gt;,&lt;/span&gt; &lt;span class="n"&gt;animal2&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Then I can add a &lt;code&gt;Cat&lt;/code&gt; type and a &lt;code&gt;Dog&lt;/code&gt; type object to it as well:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;cat1&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;Cat&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="s"&gt;"cat1"&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt;
&lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;dog1&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;Dog&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="s"&gt;"dog1"&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt;
&lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;allTypesOfAnimals&lt;/span&gt; &lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;List&lt;/span&gt;&lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="kt"&gt;Animal&lt;/span&gt;&lt;span class="o"&gt;]&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="n"&gt;cat1&lt;/span&gt; &lt;span class="o"&gt;::&lt;/span&gt; &lt;span class="n"&gt;dog1&lt;/span&gt; &lt;span class="o"&gt;::&lt;/span&gt; &lt;span class="n"&gt;animals&lt;/span&gt; &lt;span class="c1"&gt;// this works as expected&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;But what about the reverse case? Does this relation hold true when the directions are reversed? &lt;br&gt;
Let's try passing an object of type &lt;code&gt;A&lt;/code&gt; where &lt;code&gt;B&lt;/code&gt; is expected when &lt;code&gt;A :&amp;gt; B&lt;/code&gt;:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;cats&lt;/span&gt; &lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;List&lt;/span&gt;&lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="kt"&gt;Cat&lt;/span&gt;&lt;span class="o"&gt;]&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="nc"&gt;List&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;cat1&lt;/span&gt;&lt;span class="o"&gt;,&lt;/span&gt; &lt;span class="n"&gt;cat2&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt;
&lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;brokenCats&lt;/span&gt; &lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;List&lt;/span&gt;&lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="kt"&gt;Cat&lt;/span&gt;&lt;span class="o"&gt;]&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="n"&gt;animal1&lt;/span&gt; &lt;span class="o"&gt;::&lt;/span&gt; &lt;span class="n"&gt;cats&lt;/span&gt; &lt;span class="c1"&gt;// error&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Using the above example, it becomes clear that while we can add a &lt;code&gt;Cat&lt;/code&gt; to a &lt;code&gt;List[Animal]&lt;/code&gt;, we cannot add an &lt;code&gt;Animal&lt;/code&gt; to a &lt;code&gt;List[Cat]&lt;/code&gt;. The intuition behind this is that a subtype is special in some sense, and a supertype cannot replace it. Here, if we were to successfully pass &lt;code&gt;Animal&lt;/code&gt; to a &lt;code&gt;List[Cat]&lt;/code&gt;, it would break in case we call &lt;code&gt;catonly()&lt;/code&gt; on &lt;code&gt;List[Cat]&lt;/code&gt;. &lt;/p&gt;

&lt;p&gt;Working in OOP based languages makes this sound kinda obvious as we think about the same concept it in terms of Inheritance: an object of subclass cannot be passed where an object of superclass is required.&lt;/p&gt;

&lt;h3&gt;
  
  
  Subtyping v/s Inheritance
&lt;/h3&gt;

&lt;p&gt;At this point you may ask me, "Amita, why are you confusing us with subtyping and inheritance? Aren't we talking about inheritance here?" Well, actually, yes and no. I found this article &lt;a href="https://www.cmi.ac.in/~madhavan/courses/pl2009/lecturenotes/lecture-notes/node28.html#:~:text=In%20the%20object%2Doriented%20framework,refers%20to%20compatibility%20of%20interfaces.&amp;amp;text=Inheritance%20refers%20to%20reuse%20of%20implementations."&gt;here&lt;/a&gt; that can help:&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;Subtyping refers to compatibility of interfaces. A type &lt;code&gt;B&lt;/code&gt; is a subtype of &lt;code&gt;A&lt;/code&gt; if every function that can be invoked on an object of type &lt;code&gt;A&lt;/code&gt; can also be invoked on an object of type &lt;code&gt;B&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;Inheritance refers to reuse of implementations. A type &lt;code&gt;B&lt;/code&gt; inherits from another type &lt;code&gt;A&lt;/code&gt; if some functions for &lt;code&gt;B&lt;/code&gt; are written in terms of functions of &lt;code&gt;A&lt;/code&gt;.&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;In OOP, inheritance and subtyping usually go hand in hand, the syntax being the same makes it more confusing. For example, in our case, &lt;code&gt;Cat&lt;/code&gt; and &lt;code&gt;Dog&lt;/code&gt; are subtypes of &lt;code&gt;Animal&lt;/code&gt;: any operations done on &lt;code&gt;Animal&lt;/code&gt; can be done on &lt;code&gt;Cat&lt;/code&gt; and &lt;code&gt;Dog&lt;/code&gt; as well. Also, &lt;code&gt;Cat&lt;/code&gt; and &lt;code&gt;Dog&lt;/code&gt; inherit from &lt;code&gt;Animal&lt;/code&gt;: we have redefined (overridden) the function in &lt;code&gt;Animal&lt;/code&gt; in &lt;code&gt;Dog&lt;/code&gt; and &lt;code&gt;Cat&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;Inheritance helps you reuse the implementations  of methods/instance variables inside the super class, on the other hand, Subtyping deals with the safe behaviour of this class (a type) on the whole when it is passed to another data structure or function (formally called Variance, which we talk about later).&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;Subclassing doesn't guarantee Subtyping.&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;When we talk about subtyping here, we don't discuss reusing implementations, but actually delve into the behaviours of types in different conditions.&lt;/p&gt;

&lt;h2&gt;
  
  
  Variance -  Subtyping for complex types
&lt;/h2&gt;

&lt;p&gt;Now that we have established how subtypes and supertypes are used and how they behave, let's take it a bit further. What about complex types? &lt;/p&gt;

&lt;h3&gt;
  
  
  Complex Types
&lt;/h3&gt;

&lt;p&gt;Complex types are types that are composed of other types. e.g. &lt;code&gt;List&lt;/code&gt;s, &lt;code&gt;Map&lt;/code&gt;s, &lt;code&gt;Option&lt;/code&gt;s, &lt;code&gt;Function&lt;/code&gt;s etc... A &lt;code&gt;List&lt;/code&gt; alone has no meaning. They are to be used as a list of some type, say &lt;code&gt;List[Integer]&lt;/code&gt;, &lt;code&gt;List[Animal]&lt;/code&gt; etc. Similarly, a &lt;code&gt;function&lt;/code&gt; has a type signature, say, &lt;code&gt;func (a : Integer, b : String) : Boolean&lt;/code&gt;. Here a function &lt;code&gt;func&lt;/code&gt; has the type signature as &lt;code&gt;(Integer -&amp;gt; String) -&amp;gt; Boolean&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;Variance defined formally is &lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;Variance is the correlation of subtyping relationships of complex types and the subtyping relationships of their component types.&lt;/p&gt;
&lt;/blockquote&gt;

&lt;h3&gt;
  
  
  Subtyping with Lists: An instance of Covariance
&lt;/h3&gt;

&lt;p&gt;We are going to continue with our example above to understand this concept step by step. Let's take a complex type, a &lt;code&gt;List&lt;/code&gt;. If we have types &lt;code&gt;A&lt;/code&gt; and &lt;code&gt;B&lt;/code&gt; such that &lt;code&gt;A:&amp;gt;B&lt;/code&gt;, how do we relate &lt;code&gt;List[A]&lt;/code&gt; to a &lt;code&gt;List[B]&lt;/code&gt;?&lt;/p&gt;

&lt;p&gt;Reiterating on what we established above, if &lt;code&gt;A:&amp;gt;B&lt;/code&gt;, we can pass &lt;code&gt;B&lt;/code&gt; wherever we expect &lt;code&gt;A&lt;/code&gt;. &lt;/p&gt;

&lt;p&gt;Subtyping on complex types gives rise to 2 cases: &lt;/p&gt;

&lt;h4&gt;
  
  
  Case 1: pass &lt;code&gt;List[B]&lt;/code&gt; where we expect &lt;code&gt;List[A]&lt;/code&gt;
&lt;/h4&gt;

&lt;p&gt;I have a function that expects a &lt;code&gt;List[Animal]&lt;/code&gt; and does something with it:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;expectingListOfSupertype&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;animals&lt;/span&gt; &lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;List&lt;/span&gt;&lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="kt"&gt;Animal&lt;/span&gt;&lt;span class="o"&gt;])&lt;/span&gt;&lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;Unit&lt;/span&gt; &lt;span class="o"&gt;={&lt;/span&gt;
  &lt;span class="nf"&gt;println&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="nv"&gt;animals&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="py"&gt;map&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="nv"&gt;_&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="py"&gt;toString&lt;/span&gt;&lt;span class="o"&gt;))&lt;/span&gt;
&lt;span class="o"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Can I pass &lt;code&gt;cats: List[Cat]&lt;/code&gt; in place of &lt;code&gt;animals: List[Animal]&lt;/code&gt;?&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="nf"&gt;expectingListOfSupertype&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;cats&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt; &lt;span class="c1"&gt;// yes, I can!&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h4&gt;
  
  
  Case 2: pass &lt;code&gt;List[A]&lt;/code&gt; where we expect &lt;code&gt;List[B]&lt;/code&gt;
&lt;/h4&gt;

&lt;p&gt;This is the converse case to case 1.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;expectingListOfSubtype&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;cats&lt;/span&gt;&lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;List&lt;/span&gt;&lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="kt"&gt;Cat&lt;/span&gt;&lt;span class="o"&gt;])&lt;/span&gt;&lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;Unit&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="o"&gt;{&lt;/span&gt;
  &lt;span class="nf"&gt;println&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="nv"&gt;cats&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="py"&gt;map&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="nv"&gt;_&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="py"&gt;catonly&lt;/span&gt;&lt;span class="o"&gt;))&lt;/span&gt;
&lt;span class="o"&gt;}&lt;/span&gt;

&lt;span class="nf"&gt;expectingListOfSubtype&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;animals&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt; &lt;span class="c1"&gt;//type mismatch compile time error&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;No, I can't. This makes sense because if we could pass &lt;code&gt;List[Animal]&lt;/code&gt; to the function where &lt;code&gt;List[Cat]&lt;/code&gt; is expected, it could fail on calling the specific function &lt;code&gt;catonly&lt;/code&gt; on an animal instance. &lt;/p&gt;

&lt;p&gt;Hence, in case of Lists, we can substitute &lt;code&gt;List[B]&lt;/code&gt; in place of &lt;code&gt;List[B]&lt;/code&gt; when &lt;code&gt;A:&amp;gt;B&lt;/code&gt; but not the opposite.&lt;/p&gt;

&lt;p&gt;This proves a direct relationship: &lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;if &lt;code&gt;A:&amp;gt;B&lt;/code&gt;, then &lt;code&gt;List[A]:&amp;gt;List[B]&lt;/code&gt;&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;The above phenomenon is called &lt;strong&gt;Covariance&lt;/strong&gt;.&lt;/p&gt;

&lt;h3&gt;
  
  
  Subtyping with Functions: An instance of Contravariance
&lt;/h3&gt;

&lt;p&gt;Let's take another example of a complex type &lt;code&gt;Function&lt;/code&gt;. If we have types &lt;code&gt;A&lt;/code&gt; and &lt;code&gt;B&lt;/code&gt; such that &lt;code&gt;A:&amp;gt;B&lt;/code&gt;, how do we relate &lt;code&gt;Function[A]&lt;/code&gt; to a &lt;code&gt;Function[B]&lt;/code&gt;?&lt;/p&gt;

&lt;p&gt;Reiterating our subtyping relation of component types &lt;code&gt;A&lt;/code&gt; and &lt;code&gt;B&lt;/code&gt;, if &lt;code&gt;A:&amp;gt;B&lt;/code&gt;, we can pass &lt;code&gt;B&lt;/code&gt; wherever we expect &lt;code&gt;A&lt;/code&gt;. Can this relation hold true for complex type Functions as well?&lt;/p&gt;

&lt;p&gt;Let's define a function with supertype, and another function that calls this function.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;superTypeFunction&lt;/span&gt; &lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;Animal&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="nc"&gt;String&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="n"&gt;animal&lt;/span&gt; &lt;span class="k"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="n"&gt;s&lt;/span&gt;&lt;span class="s"&gt;"${animal.name} is here"&lt;/span&gt;

&lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;functionThatExpectsSupertypeFunction&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;fun&lt;/span&gt;&lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;Animal&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="nc"&gt;String&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="o"&gt;{&lt;/span&gt;
    &lt;span class="c1"&gt;// calls fun somewhere&lt;/span&gt;
&lt;span class="o"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Similarly, we define a function with subtype, and another function that calls this function&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;subTypeFunction&lt;/span&gt; &lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;Cat&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="nc"&gt;String&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="n"&gt;cat&lt;/span&gt; &lt;span class="k"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="n"&gt;s&lt;/span&gt;&lt;span class="s"&gt;"${cat.name} cat is here and has special ${cat.catonly} behaviour"&lt;/span&gt;

&lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;functionThatExpectsSubtypeFunction&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;fun&lt;/span&gt;&lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;Cat&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="nc"&gt;String&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="o"&gt;{&lt;/span&gt;
    &lt;span class="c1"&gt;// calls fun somewhere&lt;/span&gt;
&lt;span class="o"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;How are the functions going to behave in the following cases?&lt;/p&gt;

&lt;h4&gt;
  
  
  Case1: pass f[B] where we expect f[A]
&lt;/h4&gt;

&lt;p&gt;Lets try passing the subtype function to &lt;code&gt;functionThatExpectsSupertypeFunction&lt;/code&gt;:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="nf"&gt;functionThatExpectsSupertypeFunction&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;subTypeFunction&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt; &lt;span class="c1"&gt;// type mismatch!&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;The above scenario in case of functions doesn't compile. But why?&lt;/p&gt;

&lt;p&gt;Let's consider the below implementation of &lt;code&gt;functionThatExpectsSupertypeFunction&lt;/code&gt;:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;functionThatExpectsSupertypeFunction&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;fun&lt;/span&gt;&lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;Animal&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="nc"&gt;String&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="o"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;fish&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;Animal&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="s"&gt;"nemo"&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt;
  &lt;span class="nf"&gt;println&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="nf"&gt;fun&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;fish&lt;/span&gt;&lt;span class="o"&gt;))&lt;/span&gt;
&lt;span class="o"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;If the compiler could allow passing subtype function to the above, it would break coz internally it would call &lt;code&gt;fun: Cat =&amp;gt; String&lt;/code&gt; which can then try to call &lt;code&gt;catonly&lt;/code&gt; on an &lt;code&gt;Animal&lt;/code&gt;. Ouch!&lt;/p&gt;

&lt;h4&gt;
  
  
  Case2: pass f[A] where we expect f[B]
&lt;/h4&gt;

&lt;p&gt;Now let's try passing supertype function to &lt;code&gt;functionThatExpectsSubtypeFunction&lt;/code&gt;:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="nf"&gt;functionThatExpectsSubtypeFunction&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;superTypeFunction&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt; &lt;span class="c1"&gt;// compiles!&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;On the other hand, we can pass a function of supertype where subtype is expected.&lt;/p&gt;

&lt;p&gt;The intuition can be understood by thinking as to how this works in an example implementation. Suppose I implement &lt;code&gt;functionThatExpectsSubtypeFunction&lt;/code&gt; like this:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;functionThatExpectsSubtypeFunction&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;fun&lt;/span&gt;&lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;Cat&lt;/span&gt; &lt;span class="o"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="nc"&gt;String&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="o"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;cat&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="k"&gt;new&lt;/span&gt; &lt;span class="nc"&gt;Cat&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="s"&gt;"kitty"&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt;
  &lt;span class="nf"&gt;println&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="nf"&gt;fun&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;cat&lt;/span&gt;&lt;span class="o"&gt;))&lt;/span&gt;
&lt;span class="o"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;In the above scenario, even if we pass an &lt;code&gt;Animal =&amp;gt; String&lt;/code&gt; function to  &lt;code&gt;fun&lt;/code&gt; parameter, it always supplies the subtype is supplied to it. And by definition of subtyping, &lt;code&gt;Cat&lt;/code&gt; can be used wherever &lt;code&gt;Animal&lt;/code&gt; is expected. Hence this works!&lt;/p&gt;

&lt;p&gt;You can rightly observe, that &lt;code&gt;Function&lt;/code&gt;s have the opposite subtyping behaviour as compared to &lt;code&gt;List&lt;/code&gt;s. &lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;if &lt;code&gt;A:&amp;gt;B&lt;/code&gt;, then &lt;code&gt;Function[B]:&amp;gt;Function[A]&lt;/code&gt;&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;This is called &lt;strong&gt;Contravariance&lt;/strong&gt;.&lt;/p&gt;

&lt;h3&gt;
  
  
  Subtyping with Arrays: An instance of Invariance
&lt;/h3&gt;

&lt;p&gt;Let's talk about another data structure: &lt;code&gt;Arrays&lt;/code&gt;. These are mutable data structures, so for an &lt;code&gt;Array[Animal]&lt;/code&gt; we can write &lt;code&gt;Array[Cat]&lt;/code&gt; anytime. At the same time while reading we can expect &lt;code&gt;Array[i]&lt;/code&gt; can be an instance of &lt;code&gt;Animal&lt;/code&gt;, &lt;code&gt;Cat&lt;/code&gt; or &lt;code&gt;Dog&lt;/code&gt;...&lt;/p&gt;

&lt;p&gt;Thus, we can say that it can neither be safe for arrays to be Covariant or Contravariant. Confused? Wikipedia explains it better:&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;If we wish to avoid type errors, then only the third choice is safe. Clearly, not every &lt;code&gt;Animal[]&lt;/code&gt; can be treated as if it were a &lt;code&gt;Cat[]&lt;/code&gt;, since a client reading from the array will expect a &lt;code&gt;Cat&lt;/code&gt;, but an &lt;code&gt;Animal[]&lt;/code&gt; may contain e.g. a &lt;code&gt;Dog&lt;/code&gt;. So the contravariant rule is not safe.&lt;/p&gt;

&lt;p&gt;Conversely, a &lt;code&gt;Cat[]&lt;/code&gt; cannot be treated as an &lt;code&gt;Animal[]&lt;/code&gt;. It should always be possible to put a &lt;code&gt;Dog&lt;/code&gt; into an &lt;code&gt;Animal[]&lt;/code&gt;. With covariant arrays this cannot be guaranteed to be safe, since the backing store might actually be an array of cats. So the covariant rule is also not safe—the array constructor should be invariant.&lt;br&gt;
&lt;/p&gt;
&lt;/blockquote&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight scala"&gt;&lt;code&gt;&lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;animalarray&lt;/span&gt;&lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;Array&lt;/span&gt;&lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="kt"&gt;Animal&lt;/span&gt;&lt;span class="o"&gt;]&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="nc"&gt;Array&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;animal1&lt;/span&gt;&lt;span class="o"&gt;,&lt;/span&gt; &lt;span class="n"&gt;animal2&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt;
&lt;span class="k"&gt;val&lt;/span&gt; &lt;span class="nv"&gt;catarray&lt;/span&gt;&lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;Array&lt;/span&gt;&lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="kt"&gt;Cat&lt;/span&gt;&lt;span class="o"&gt;]&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="nc"&gt;Array&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;cat1&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt;

&lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;expectingArrayOfSupertype&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;animals&lt;/span&gt;&lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;Array&lt;/span&gt;&lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="kt"&gt;Animal&lt;/span&gt;&lt;span class="o"&gt;])&lt;/span&gt;&lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="o"&gt;{&lt;/span&gt;
  &lt;span class="c1"&gt;// some read/write operations&lt;/span&gt;
&lt;span class="o"&gt;}&lt;/span&gt;
&lt;span class="k"&gt;def&lt;/span&gt; &lt;span class="nf"&gt;expectingArrayOfSubtype&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;cats&lt;/span&gt;&lt;span class="k"&gt;:&lt;/span&gt; &lt;span class="kt"&gt;Array&lt;/span&gt;&lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="kt"&gt;Cat&lt;/span&gt;&lt;span class="o"&gt;])&lt;/span&gt; &lt;span class="k"&gt;=&lt;/span&gt; &lt;span class="o"&gt;{&lt;/span&gt;
  &lt;span class="c1"&gt;// some read/write operations&lt;/span&gt;
&lt;span class="o"&gt;}&lt;/span&gt;

&lt;span class="nf"&gt;expectingArrayOfSupertype&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;catarray&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt; &lt;span class="c1"&gt;// type mismatch!!&lt;/span&gt;
&lt;span class="nf"&gt;expectingArrayOfSubtype&lt;/span&gt;&lt;span class="o"&gt;(&lt;/span&gt;&lt;span class="n"&gt;animalarray&lt;/span&gt;&lt;span class="o"&gt;)&lt;/span&gt; &lt;span class="c1"&gt;// type mismatch!!&lt;/span&gt;

&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Both our cases fail for Arrays, they being &lt;strong&gt;Invariant&lt;/strong&gt;.&lt;/p&gt;

&lt;h4&gt;
  
  
  Special case of Variance of Arrays in Java
&lt;/h4&gt;

&lt;p&gt;For Java / C# developers, it is worth noting that Arrays are covariant. The reason is in detail &lt;a href="https://en.wikipedia.org/wiki/Covariance_and_contravariance_(computer_science)#Covariant_arrays_in_Java_and_C#"&gt;here&lt;/a&gt;. Hint: it's because Generics was included much later in the language than Arrays...&lt;/p&gt;

&lt;h2&gt;
  
  
  Rule of Thumb
&lt;/h2&gt;

&lt;p&gt;Well, at this point it can be too much to digest. But there can be a simple rule of thumb to it:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Read-only data types/sources can be covariant, e.g. immutable &lt;code&gt;List&lt;/code&gt;s&lt;/li&gt;
&lt;li&gt;write-only data types/sinks can be contravariant, e.g. functions&lt;/li&gt;
&lt;li&gt;Mutable data types which act as both sources and sinks should be invariant. e.g. &lt;code&gt;Array&lt;/code&gt;s&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Contravariance is quite unintuitive in nature, as it is difficult to imagine a type with just write-only capabilities. In case of functions(the case where we're only talking about it &lt;em&gt;accepting&lt;/em&gt; a type &lt;em&gt;once&lt;/em&gt;), we can think that it took a subtype, but did not return it, hence making it 'write-only'. Usually, a complex type that accepts a simple type and simply consumes it, can be treated as contravariant.&lt;/p&gt;

&lt;h2&gt;
  
  
  Try this code
&lt;/h2&gt;

&lt;p&gt;This whole post makes more sense on running the code snippets. Checkout &lt;a href="https://gist.github.com/amita-shukla/32f14cf325aa89296b10297862fc198f"&gt;this&lt;/a&gt; gist which contains the entire code. I would recommend running them one by one and observe the magic of types yourself!&lt;/p&gt;

&lt;h2&gt;
  
  
  Further Reading
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;This &lt;a href="https://en.wikipedia.org/wiki/Covariance_and_contravariance_(computer_science)"&gt;Wikipedia article&lt;/a&gt; is my starting point, and the &lt;code&gt;Animal&lt;/code&gt; example is inspired from here.&lt;/li&gt;
&lt;li&gt;&lt;a href="https://en.wikipedia.org/wiki/Liskov_substitution_principle"&gt;Liskov's Substitution Principle&lt;/a&gt;&lt;/li&gt;
&lt;li&gt;&lt;a href="https://www.cs.princeton.edu/courses/archive/fall98/cs441/mainus/node12.html"&gt;SubTypes v/s Subclasses&lt;/a&gt;&lt;/li&gt;
&lt;li&gt;&lt;a href="https://www.cmi.ac.in/~madhavan/courses/pl2009/lecturenotes/lecture-notes/node28.html"&gt;Subtyping vs Inheritance&lt;/a&gt;&lt;/li&gt;
&lt;li&gt;&lt;a href="https://docs.scala-lang.org/tour/variances.html"&gt;Scala Docs: Variance&lt;/a&gt;&lt;/li&gt;
&lt;/ul&gt;

</description>
      <category>programming</category>
      <category>variance</category>
      <category>subtypes</category>
      <category>scala</category>
    </item>
  </channel>
</rss>
