Generics #
A major part of software engineering is building components that not only have well-defined and consistent APIs, but are also reusable. Components that are capable of working on the data of today as well as the data of tomorrow will give you the most flexible capabilities for building up large software systems.
In languages like C# and Java, one of the main tools in the toolbox for creating reusable components is generics, that is, being able to create a component that can work over a variety of types rather than a single one. This allows users to consume these components and use their own types.
Hello World of Generics #
To start off, let’s do the “hello world” of generics: the identity
function. The identity function is a function that will return back
whatever is passed in. You can think of this in a similar way to the
echo
command.
Without generics, we would either have to give the identity function a specific type:
function identity(arg: number): number {
return arg;
}
Or, we could describe the identity function using the any
type:
function identity(arg: any): any {
return arg;
}
While using any
is certainly generic in that it will cause the
function to accept any and all types for the type of arg
, we actually
are losing the information about what that type was when the function
returns. If we passed in a number, the only information we have is that
any type could be returned.
Instead, we need a way of capturing the type of the argument in such a way that we can also use it to denote what is being returned. Here, we will use a type variable, a special kind of variable that works on types rather than values.
function identity<Type>(arg: Type): Type {
return arg;
}
We’ve now added a type variable Type
to the identity function. This
Type
allows us to capture the type the user provides (e.g. number
),
so that we can use that information later. Here, we use Type
again as
the return type. On inspection, we can now see the same type is used for
the argument and the return type. This allows us to traffic that type
information in one side of the function and out the other.
We say that this version of the identity
function is generic, as it
works over a range of types. Unlike using any
, it’s also just as
precise (i.e., it doesn’t lose any information) as the first identity
function that used numbers for the argument and return type.
Once we’ve written the generic identity function, we can call it in one of two ways. The first way is to pass all of the arguments, including the type argument, to the function:
let output = identity<string>("myString");
Here we explicitly set Type
to be string
as one of the arguments to
the function call, denoted using the <>
around the arguments rather
than ()
.
The second way is also perhaps the most common. Here we use type
argument inference — that is, we want the compiler to set the value
of Type
for us automatically based on the type of the argument we pass
in:
let output = identity("myString");
Notice that we didn’t have to explicitly pass the type in the angle
brackets (<>
); the compiler just looked at the value "myString"
, and
set Type
to its type. While type argument inference can be a helpful
tool to keep code shorter and more readable, you may need to explicitly
pass in the type arguments as we did in the previous example when the
compiler fails to infer the type, as may happen in more complex
examples.
Working with Generic Type Variables #
When you begin to use generics, you’ll notice that when you create
generic functions like identity
, the compiler will enforce that you
use any generically typed parameters in the body of the function
correctly. That is, that you actually treat these parameters as if they
could be any and all types.
Let’s take our identity
function from earlier:
function identity<Type>(arg: Type): Type {
return arg;
}
What if we want to also log the length of the argument arg
to the
console with each call? We might be tempted to write this:
function loggingIdentity<Type>(arg: Type): Type {
console.log(arg.length);
return arg;
}
When we do, the compiler will give us an error that we’re using the
.length
member of arg
, but nowhere have we said that arg
has this
member. Remember, we said earlier that these type variables stand in for
any and all types, so someone using this function could have passed in a
number
instead, which does not have a .length
member.
Let’s say that we’ve actually intended this function to work on arrays
of Type
rather than Type
directly. Since we’re working with arrays,
the .length
member should be available. We can describe this just like
we would create arrays of other types:
function loggingIdentity<Type>(arg: Type[]): Type[] {
console.log(arg.length);
return arg;
}
You can read the type of loggingIdentity
as “the generic function
loggingIdentity
takes a type parameter Type
, and an argument arg
which is an array of Type
s, and returns an array of Type
s.” If we
passed in an array of numbers, we’d get an array of numbers back out, as
Type
would bind to number
. This allows us to use our generic type
variable Type
as part of the types we’re working with, rather than the
whole type, giving us greater flexibility.
We can alternatively write the sample example this way:
function loggingIdentity<Type>(arg: Array<Type>): Array<Type> {
console.log(arg.length); // Array has a .length, so no more error
return arg;
}
You may already be familiar with this style of type from other
languages. In the next section, we’ll cover how you can create your own
generic types like Array<Type>
.
Generic Types #
In previous sections, we created generic identity functions that worked over a range of types. In this section, we’ll explore the type of the functions themselves and how to create generic interfaces.
The type of generic functions is just like those of non-generic functions, with the type parameters listed first, similarly to function declarations:
function identity<Type>(arg: Type): Type {
return arg;
}
Â
let myIdentity: <Type>(arg: Type) => Type = identity;
We could also have used a different name for the generic type parameter in the type, so long as the number of type variables and how the type variables are used line up.
function identity<Input>(arg: Input): Input {
return arg;
}
Â
let myIdentity: <Input>(arg: Input) => Input = identity;
We can also write the generic type as a call signature of an object literal type:
function identity<Type>(arg: Type): Type {
return arg;
}
Â
let myIdentity: { <Type>(arg: Type): Type } = identity;
Which leads us to writing our first generic interface. Let’s take the object literal from the previous example and move it to an interface:
interface GenericIdentityFn {
<Type>(arg: Type): Type;
}
Â
function identity<Type>(arg: Type): Type {
return arg;
}
Â
let myIdentity: GenericIdentityFn = identity;
In a similar example, we may want to move the generic parameter to be a
parameter of the whole interface. This lets us see what type(s) we’re
generic over (e.g. Dictionary<string>
rather than just Dictionary
).
This makes the type parameter visible to all the other members of the
interface.
interface GenericIdentityFn<Type> {
(arg: Type): Type;
}
Â
function identity<Type>(arg: Type): Type {
return arg;
}
Â
let myIdentity: GenericIdentityFn<number> = identity;
Notice that our example has changed to be something slightly different.
Instead of describing a generic function, we now have a non-generic
function signature that is a part of a generic type. When we use
GenericIdentityFn
, we now will also need to specify the corresponding
type argument (here: number
), effectively locking in what the
underlying call signature will use. Understanding when to put the type
parameter directly on the call signature and when to put it on the
interface itself will be helpful in describing what aspects of a type
are generic.
In addition to generic interfaces, we can also create generic classes. Note that it is not possible to create generic enums and namespaces.
Generic Classes #
A generic class has a similar shape to a generic interface. Generic
classes have a generic type parameter list in angle brackets (<>
)
following the name of the class.
class GenericNumber<NumType> {
zeroValue: NumType;
add: (x: NumType, y: NumType) => NumType;
}
Â
let myGenericNumber = new GenericNumber<number>();
myGenericNumber.zeroValue = 0;
myGenericNumber.add = function (x, y) {
return x + y;
};
This is a pretty literal use of the GenericNumber
class, but you may
have noticed that nothing is restricting it to only use the number
type. We could have instead used string
or even more complex objects.
let stringNumeric = new GenericNumber<string>();
stringNumeric.zeroValue = "";
stringNumeric.add = function (x, y) {
return x + y;
};
Â
console.log(stringNumeric.add(stringNumeric.zeroValue, "test"));
Just as with interface, putting the type parameter on the class itself lets us make sure all of the properties of the class are working with the same type.
As we cover in our section on classes, a class has two sides to its type: the static side and the instance side. Generic classes are only generic over their instance side rather than their static side, so when working with classes, static members can not use the class’s type parameter.
Generic Constraints #
If you remember from an earlier example, you may sometimes want to write
a generic function that works on a set of types where you have some
knowledge about what capabilities that set of types will have. In our
loggingIdentity
example, we wanted to be able to access the .length
property of arg
, but the compiler could not prove that every type had
a .length
property, so it warns us that we can’t make this assumption.
function loggingIdentity<Type>(arg: Type): Type {
console.log(arg.length);
return arg;
}
Instead of working with any and all types, we’d like to constrain this
function to work with any and all types that also have the .length
property. As long as the type has this member, we’ll allow it, but it’s
required to have at least this member. To do so, we must list our
requirement as a constraint on what Type
can be.
To do so, we’ll create an interface that describes our constraint. Here,
we’ll create an interface that has a single .length
property and then
we’ll use this interface and the extends
keyword to denote our
constraint:
interface Lengthwise {
length: number;
}
Â
function loggingIdentity<Type extends Lengthwise>(arg: Type): Type {
console.log(arg.length); // Now we know it has a .length property, so no more error
return arg;
}
Because the generic function is now constrained, it will no longer work over any and all types:
loggingIdentity(3);
Instead, we need to pass in values whose type has all the required properties:
loggingIdentity({ length: 10, value: 3 });
Using Type Parameters in Generic Constraints #
You can declare a type parameter that is constrained by another type
parameter. For example, here we’d like to get a property from an object
given its name. We’d like to ensure that we’re not accidentally grabbing
a property that does not exist on the obj
, so we’ll place a constraint
between the two types:
function getProperty<Type, Key extends keyof Type>(obj: Type, key: Key) {
return obj[key];
}
Â
let x = { a: 1, b: 2, c: 3, d: 4 };
Â
getProperty(x, "a");
getProperty(x, "m");
Using Class Types in Generics #
When creating factories in TypeScript using generics, it is necessary to refer to class types by their constructor functions. For example,
function create<Type>(c: { new (): Type }): Type {
return new c();
}
A more advanced example uses the prototype property to infer and constrain relationships between the constructor function and the instance side of class types.
class BeeKeeper {
hasMask: boolean = true;
}
Â
class ZooKeeper {
nametag: string = "Mikle";
}
Â
class Animal {
numLegs: number = 4;
}
Â
class Bee extends Animal {
numLegs = 6;
keeper: BeeKeeper = new BeeKeeper();
}
Â
class Lion extends Animal {
keeper: ZooKeeper = new ZooKeeper();
}
Â
function createInstance<A extends Animal>(c: new () => A): A {
return new c();
}
Â
createInstance(Lion).keeper.nametag;
createInstance(Bee).keeper.hasMask;
This pattern is used to power the mixins design pattern.
Generic Parameter Defaults #
Consider a function that creates a new HTMLElement
. Calling the
function with no arguments generates a Div
; calling it with an element
as the first argument generates an element of the argument’s type. You
can optionally pass a list of children as well. Previously you would
have to define it as:
declare function create(): Container<HTMLDivElement, HTMLDivElement[]>;
declare function create<T extends HTMLElement>(element: T): Container<T, T[]>;
declare function create<T extends HTMLElement, U extends HTMLElement>(
element: T,
children: U[]
): Container<T, U[]>;
With generic parameter defaults we can reduce it to:
declare function create<T extends HTMLElement = HTMLDivElement, U = T[]>(
element?: T,
children?: U
): Container<T, U>;
A generic parameter default follows the following rules:
- A type parameter is deemed optional if it has a default.
- Required type parameters must not follow optional type parameters.
- Default types for a type parameter must satisfy the constraint for the type parameter, if it exists.
- When specifying type arguments, you are only required to specify type arguments for the required type parameters. Unspecified type parameters will resolve to their default types.
- If a default type is specified and inference cannot choose a candidate, the default type is inferred.
- A class or interface declaration that merges with an existing class or interface declaration may introduce a default for an existing type parameter.
- A class or interface declaration that merges with an existing class or interface declaration may introduce a new type parameter as long as it specifies a default.
::: _attribution
© 2012-2023 Microsoft
Licensed under the Apache License, Version 2.0.
https://www.typescriptlang.org/docs/handbook/2/generics.html{._attribution-link}
:::