![]() | This article provides insufficient context for those unfamiliar with the subject.(October 2018) |
In programming language type theory, row polymorphism is a kind of polymorphism that allows one to write programs that are structurally [1] (rather than nominally) polymorphic on record types and/or variants. [2]
A row-polymorphic type system and proof of type inference for records was introduced by Mitchell Wand. [3] [4]
The theoretical treatment of row polymorphism is somewhat complicated by the need to have distinct labels in a record. One approach, taken by Rémy and colleagues (and implicitly present in the presentation below, which is heavily inspired by Wand's), is to consider different kinds of row types, depending on their labels. [2]
Gaster and Jones unformely extended the approach to variants as well, by making both the record and variant type constructor map from row kinds to types. [5] Blume et al. proposed to take the approach one step further into the realm of composable extensions by adding "first-class cases" handled by a LetCC
construct to compose continuations in the case-matching code. [6]
Another approach to the issue of unique labels is to extend system F with a "coherent merge" operator, resulting in calculi with so-called disjoint intersection types. [7]
Morris and McKinna generalized row types to row theories to uniformly handle varying notions of (record) extension in a single theoretical framework: for instance one application may desire the operation of extension to overwrite existing fields in case of matching names, another may want to keep both of them accessible, possibly some path-based addressing scheme etc. [8]
The row-polymorphic record type defines a list of fields with their corresponding types, a list of missing fields, and a variable indicating the absence or presence of arbitrary additional fields. Both lists are optional, and the variable may be constrained. Specifically, the variable may be "empty", indicating that no additional fields may be present for the record.
It may be written as . This indicates a record type that has fields with respective types of (for ), and does not have any of the fields (for ), while expresses the fact the record may contain other fields than .
Row-polymorphic record types allow us to write programs that operate only on a section of a record. For example, one may define a function that performs some two-dimensional transformation that accepts a record with two or more coordinates, and returns an identical type:
Thanks to row polymorphism, the function may perform two-dimensional transformation on a three-dimensional (in fact, n-dimensional) point, leaving the z coordinate (or any other coordinates) intact. In a more general sense, the function can perform on any record that contains the fields and with type . There is no loss of information: the type ensures that all the fields represented by the variable are present in the return type. In contrast, the type definition expresses the fact that a record of that type has exactly the and fields and nothing else. In this case, a classic record type is obtained.
The record operations of selecting a field , adding a field , and removing a field can be given row-polymorphic types.
Row polymorphism is not supported in Standard ML, but is in some extensions or derivatives such as SML# [9] and Ocaml.
The very existence of SML#'s first version was motivated [10] by adding row polymorphism, based on a SIGMOD '89 paper by Ohori et al., [11] which introduced the "Machiavelli" extension to SML, although its name was changed later to "SML# of Kansai", before settling on the shorter name. The '#' in the name bears no relationship with F#, but is due to the #
operator used to implicitly define the row-polymorphic types on field access.
In Ocaml, row polymorphism is used by Ocaml's object
types and also by its 'polymorphic variants'. Ordinary record types are not row-polymorphic in Ocaml. [12]
F# can simulate row polymorphism using its "Statically Resolved Type Parameters" (SRTP) mechanism, [13] which has also been informally called "static duck typing". [14] This however is limited to inline F# functions, so that they are not exported to the .NET type system, which itself does not support such a feature.
Row polymorphism is also supported in PureScript. [15]