Generics
There is a special data type called
type
. Bindings of this value (which should be named like a type), can have any possible "type" as their value (e.g.int
,float
,int|string
,Customer
).Every binding of type
type
must have a compile time literal value.Generic types are defined using module-level functions that accept
type
arguments and return atype
.These functions must be compile time (because anything related to
type
must be) (Example A). This means that you cannot use any non-literal value as a value for a type binding. You also cannot assign a function that receives or return a type, to a function-level lambda.Note that a generic function's input of form
T|U
means caller can provide a union binding which has at least two options for the type, it may have 2 or more allowed types.If an optional generic type is omitted in a function call, compiler will infer it.
Generic functions are implemented as functions that accept
type
arguments but their output is nottype
.You can also define generic function types in the same way you define generic data structures. This can be useful when you want to separate generic parameter from non-generic ones.
Examples
Type inference
You can use T :> type
notation to indicate compiler should infer type T when function is being called.
This means:
If call is made for a non union type, then compile time of T will be inferred and used.
If call is being for a union type and there is no function to infer, compiler will infer the union type.
If call is for a union type and there are functions that need to be inferred, if we have an implementation for the union type it will be used. Otherwise, compiler will make sure we have impl for each case of the union, and delegate actual infer to runtime to use dynamic type inside the argument and use it for inference.
Polymorphism and Contracts
You can also use :>
notation when using a function of a generic type, which indicates compiler or runtime (depending on the situation) should find a correct function for you. This can be used to write different implementations of logic for each different type and let language decide which implementation to use at compile time or runtime.
A simple example:
In the above example, we define a generic type ToString
which is a function. This type can have different implementations for each type. We have specialized it for integers in intToString
function. Note that this function has explicitly set its type to ToString(int)
to tell compiler that this is an implementation of ToString for integers.
Generic functions like ToString
are called contracts because they define a behavior based on one or more types. Note that you are not limited to single typed contracts. Functions that are explicitly defined as that type are "implementations of that contract".
Below is another example of contracts:
Another example of drawing different shapes:
Above examples have all been compile time examples where compiler has all the information needed to determine what needs to be used for the contract function argument. But in case of using union bindings, this can be a runtime issue. If there is no implementation for the actual union type, then the compiler will just make sure we have implementations that can be used for all possible union options, and then delegate the actual value to be determined at runtime, based on the runtime type inside the union binding.
Note that contracts (used with dynamic union) can be used to implement solutions for expression problem. New types means new data types in the code and implementation of contracts for them. New operations, mean new contracts.
Last updated
Was this helpful?