Depth Subtyping
Assume we have two classes, which have a subtype relationship:
class Person { name: string } class Employee extends Person { department: string }
It’s valid to use an Employee
instance where a Person
instance is expected.
// @flow class Person { name: string } class Employee extends Person { department: string } var employee: Employee = new Employee; var person: Person = employee; // OK
However, it is not valid to use an object containing an Employee
instance where an object containing a Person
instance is expected.
// @flow class Person { name: string } class Employee extends Person { department: string } var employee: { who: Employee } = { who: new Employee }; // $ExpectError var person: { who: Person } = employee; // Error
This is an error because objects are mutable. The value referenced by the employee
variable is the same as the value referenced by the person
variable.
person.who = new Person;
If we write into the who
property of the person
object, we’ve also changed the value of employee.who
, which is explicitly annotated to be an Employee
instance.
If we prevented any code from ever writing a new value to the object through the person
variable, it would be safe to use the employee
variable. Flow provides a syntax for this:
// @flow class Person { name: string } class Employee extends Person { department: string } var employee: { who: Employee } = { who: new Employee }; var person: { +who: Person } = employee; // OK // $ExpectError person.who = new Person; // Error!
The plus sign indicates that the who
property is “covariant.” Using a covariant property allows us to use objects which have subtype-compatible values for that property. By default, object properties are invariant, which allow both reads and writes, but are more restrictive in the values they accept.
Read more about property variance.
© 2013–present Facebook Inc.
Licensed under the MIT License.
https://flow.org/en/docs/lang/depth-subtyping