More on Functions

More on Functions

Function Type Expressions

The simplest way to describe a function is with a function type expression. These types are syntactically similar to arrow functions:

Try this code ↗

function greeter(fn: (a: string) => void) {
fn("Hello, World");
}

functionprintToConsole(s: string) {
console.log(s);
}

greeter(printToConsole);

The syntax (a: string) => void means “a function with one parameter, named a, of type string, that doesn’t have a return value”. Just like with function declarations, if a parameter type isn’t specified, it’s implicitly any.

Note that the parameter name is required. The function type (string) => void means “a function with a parameter named string of type any“!

Of course, we can use a type alias to name a function type:

Try this code ↗

type GreetFunction = (a: string) => void;
functiongreeter(fn: GreetFunction) {
// ...
}

Call Signatures

In JavaScript, functions can have properties in addition to being callable. However, the function type expression syntax doesn’t allow for declaring properties. If we want to describe something callable with properties, we can write a call signature in an object type:

Try this code ↗

type DescribableFunction = {
description: string;
  (someArg: number): boolean;
};
functiondoSomething(fn: DescribableFunction) {
console.log(fn.description + " returned " + fn(6));
}

functionmyFunc(someArg: number) {
returnsomeArg > 3;
}
myFunc.description = "default description";

doSomething(myFunc);

Note that the syntax is slightly different compared to a function type expression - use : between the parameter list and the return type rather than =>.

Construct Signatures

JavaScript functions can also be invoked with the new operator. TypeScript refers to these as constructors because they usually create a new object. You can write a construct signature by adding the new keyword in front of a call signature:

Try this code ↗

type SomeConstructor = {
new (s: string): SomeObject;
};
functionfn(ctor: SomeConstructor) {
returnnewctor("hello");
}

Some objects, like JavaScript’s Date object, can be called with or without new. You can combine call and construct signatures in the same type arbitrarily:

Try this code ↗

interface CallOrConstruct {
  (n?: number): string;
new (s: string): Date;
}

Generic Functions

It’s common to write a function where the types of the input relate to the type of the output, or where the types of two inputs are related in some way. Let’s consider for a moment a function that returns the first element of an array:

Try this code ↗

function firstElement(arr: any[]) {
returnarr[0];
}

This function does its job, but unfortunately has the return type any. It’d be better if the function returned the type of the array element.

In TypeScript, generics are used when we want to describe a correspondence between two values. We do this by declaring a type parameter in the function signature:

Try this code ↗

function firstElement<Type>(arr: Type[]): Type | undefined {
returnarr[0];
}

By adding a type parameter Type to this function and using it in two places, we’ve created a link between the input of the function (the array) and the output (the return value). Now when we call it, a more specific type comes out:

Try this code ↗

// s is of type 'string'
consts = firstElement(["a", "b", "c"]);
// n is of type 'number'
constn = firstElement([1, 2, 3]);
// u is of type undefined
constu = firstElement([]);

Inference

Note that we didn’t have to specify Type in this sample. The type was inferred - chosen automatically - by TypeScript.

We can use multiple type parameters as well. For example, a standalone version of map would look like this:

Try this code ↗

function map<Input, Output>(arr: Input[], func: (arg: Input) => Output): Output[] {
returnarr.map(func);
}

// Parameter 'n' is of type 'string'
// 'parsed' is of type 'number[]'
constparsed = map(["1", "2", "3"], (n) =>parseInt(n));

Note that in this example, TypeScript could infer both the type of the Input type parameter (from the given string array), as well as the Output type parameter based on the return value of the function expression (number).

Constraints

We’ve written some generic functions that can work on any kind of value. Sometimes we want to relate two values, but can only operate on a certain subset of values. In this case, we can use a constraint to limit the kinds of types that a type parameter can accept.

Let’s write a function that returns the longer of two values. To do this, we need a length property that’s a number. We constrain the type parameter to that type by writing an extends clause:

Try this code ↗

function longest<Type extends { length: number }>(a: Type, b: Type) {
if (a.length >= b.length) {
returna;
  } else {
returnb;
  }
}

// longerArray is of type 'number[]'
constlongerArray = longest([1, 2], [1, 2, 3]);
// longerString is of type 'alice' | 'bob'
constlongerString = longest("alice", "bob");
// Error! Numbers don't have a 'length' property
constnotOK = longest(10, 100);
Generated error
Argument of type 'number' is not assignable to parameter of type '{ length: number; }'.

There are a few interesting things to note in this example. We allowed TypeScript to infer the return type of longest. Return type inference also works on generic functions.

Because we constrained Type to { length: number }, we were allowed to access the .length property of the a and b parameters. Without the type constraint, we wouldn’t be able to access those properties because the values might have been some other type without a length property.

The types of longerArray and longerString were inferred based on the arguments. Remember, generics are all about relating two or more values with the same type!

Finally, just as we’d like, the call to longest(10, 100) is rejected because the number type doesn’t have a .length property.

Working with Constrained Values

Here’s a common error when working with generic constraints:

Try this code ↗

function minimumLength<Type extends { length: number }>(
obj: Type,
minimum: number
): Type {
if (obj.length >= minimum) {
returnobj;
  } else {
return { length:minimum };
  }
}
Generated error
Type '{ length: number; }' is not assignable to type 'Type'.
  '{ length: number; }' is assignable to the constraint of type 'Type', but 'Type' could be instantiated with a different subtype of constraint '{ length: number; }'.

It might look like this function is OK - Type is constrained to { length: number }, and the function either returns Type or a value matching that constraint. The problem is that the function promises to return the same kind of object as was passed in, not just some object matching the constraint. If this code were legal, you could write code that definitely wouldn’t work:

Try this code ↗

// 'arr' gets value { length: 6 }
constarr = minimumLength([1, 2, 3], 6);
// and crashes here because arrays have
// a 'slice' method, but not the returned object!
console.log(arr.slice(0));

Specifying Type Arguments

TypeScript can usually infer the intended type arguments in a generic call, but not always. For example, let’s say you wrote a function to combine two arrays:

Try this code ↗

function combine<Type>(arr1: Type[], arr2: Type[]): Type[] {
returnarr1.concat(arr2);
}

Normally it would be an error to call this function with mismatched arrays:

Try this code ↗

const arr = combine([1, 2, 3], ["hello"]);
Generated error
Type 'string' is not assignable to type 'number'.

If you intended to do this, however, you could manually specify Type:

Try this code ↗

const arr = combine<string | number>([1, 2, 3], ["hello"]);

Guidelines for Writing Good Generic Functions

Writing generic functions is fun, and it can be easy to get carried away with type parameters. Having too many type parameters or using constraints where they aren’t needed can make inference less successful, frustrating callers of your function.

Push Type Parameters Down

Here are two ways of writing a function that appear similar:

Try this code ↗

function firstElement1<Type>(arr: Type[]) {
returnarr[0];
}

functionfirstElement2<Typeextendsany[]>(arr: Type) {
returnarr[0];
}

// a: number (good)
consta = firstElement1([1, 2, 3]);
// b: any (bad)
constb = firstElement2([1, 2, 3]);

These might seem identical at first glance, but firstElement1 is a much better way to write this function. Its inferred return type is Type, but firstElement2’s inferred return type is any because TypeScript has to resolve the arr[0] expression using the constraint type, rather than “waiting” to resolve the element during a call.

Rule: When possible, use the type parameter itself rather than constraining it

Use Fewer Type Parameters

Here’s another pair of similar functions:

Try this code ↗

function filter1<Type>(arr: Type[], func: (arg: Type) => boolean): Type[] {
returnarr.filter(func);
}

functionfilter2<Type, Funcextends (arg: Type) =>boolean>(
arr: Type[],
func: Func
): Type[] {
returnarr.filter(func);
}

We’ve created a type parameter Func that doesn’t relate two values. That’s always a red flag, because it means callers wanting to specify type arguments have to manually specify an extra type argument for no reason. Func doesn’t do anything but make the function harder to read and reason about!

Rule: Always use as few type parameters as possible

Type Parameters Should Appear Twice

Sometimes we forget that a function might not need to be generic:

Try this code ↗

function greet<Str extends string>(s: Str) {
console.log("Hello, " + s);
}

greet("world");

We could just as easily have written a simpler version:

Try this code ↗

function greet(s: string) {
console.log("Hello, " + s);
}

Remember, type parameters are for relating the types of multiple values. If a type parameter is only used once in the function signature, it’s not relating anything. This includes the inferred return type; for example, if Str was part of the inferred return type of greet, it would be relating the argument and return types, so would be used twice despite appearing only once in the written code.

Rule: If a type parameter only appears in one location, strongly reconsider if you actually need it

Optional Parameters

Functions in JavaScript often take a variable number of arguments. For example, the toFixed method of number takes an optional digit count:

Try this code ↗

function f(n: number) {
console.log(n.toFixed()); // 0 arguments
console.log(n.toFixed(3)); // 1 argument
}

We can model this in TypeScript by marking the parameter as optional with ?:

Try this code ↗

function f(x?: number) {
// ...
}
f(); // OK
f(10); // OK

Although the parameter is specified as type number, the x parameter will actually have the type number | undefined because unspecified parameters in JavaScript get the value undefined.

You can also provide a parameter default:

Try this code ↗

function f(x = 10) {
// ...
}

Now in the body of f, x will have type number because any undefined argument will be replaced with 10. Note that when a parameter is optional, callers can always pass undefined, as this simply simulates a “missing” argument:

Try this code ↗

declare function f(x?: number): void;
// cut
// All OK
f();
f(10);
f(undefined);

Optional Parameters in Callbacks

Once you’ve learned about optional parameters and function type expressions, it’s very easy to make the following mistakes when writing functions that invoke callbacks:

Try this code ↗

function myForEach(arr: any[], callback: (arg: any, index?: number) => void) {
for (leti = 0; i < arr.length; i++) {
callback(arr[i], i);
  }
}

What people usually intend when writing index? as an optional parameter is that they want both of these calls to be legal:

Try this code ↗

myForEach([1, 2, 3], (a) => console.log(a));
myForEach([1, 2, 3], (a, i) =>console.log(a, i));

What this actually means is that callback might get invoked with one argument. In other words, the function definition says that the implementation might look like this:

Try this code ↗

function myForEach(arr: any[], callback: (arg: any, index?: number) => void) {
for (leti = 0; i < arr.length; i++) {
// I don't feel like providing the index today
callback(arr[i]);
  }
}

In turn, TypeScript will enforce this meaning and issue errors that aren’t really possible:

Try this code ↗

myForEach([1, 2, 3], (a, i) => {
console.log(i.toFixed());
});
Generated error
'i' is possibly 'undefined'.

In JavaScript, if you call a function with more arguments than there are parameters, the extra arguments are simply ignored. TypeScript behaves the same way. Functions with fewer parameters (of the same types) can always take the place of functions with more parameters.

Rule: When writing a function type for a callback, never write an optional parameter unless you intend to call the function without passing that argument

Function Overloads

Some JavaScript functions can be called in a variety of argument counts and types. For example, you might write a function to produce a Date that takes either a timestamp (one argument) or a month/day/year specification (three arguments).

In TypeScript, we can specify a function that can be called in different ways by writing overload signatures. To do this, write some number of function signatures (usually two or more), followed by the body of the function:

Try this code ↗

function makeDate(timestamp: number): Date;
functionmakeDate(m: number, d: number, y: number): Date;
functionmakeDate(mOrTimestamp: number, d?: number, y?: number): Date {
if (d !== undefined && y !== undefined) {
returnnewDate(y, mOrTimestamp, d);
  } else {
returnnewDate(mOrTimestamp);
  }
}
constd1 = makeDate(12345678);
constd2 = makeDate(5, 5, 5);
constd3 = makeDate(1, 3);
Generated error
No overload expects 2 arguments, but overloads do exist that expect either 1 or 3 arguments.

In this example, we wrote two overloads: one accepting one argument, and another accepting three arguments. These first two signatures are called the overload signatures.

Then, we wrote a function implementation with a compatible signature. Functions have an implementation signature, but this signature can’t be called directly. Even though we wrote a function with two optional parameters after the required one, it can’t be called with two parameters!

Overload Signatures and the Implementation Signature

This is a common source of confusion. Often people will write code like this and not understand why there is an error:

Try this code ↗

function fn(x: string): void;
functionfn() {
// ...
}
// Expected to be able to call with zero arguments
fn();
Generated error
Expected 1 arguments, but got 0.

Again, the signature used to write the function body can’t be “seen” from the outside.

The signature of the implementation is not visible from the outside. When writing an overloaded function, you should always have two or more signatures above the implementation of the function.

The implementation signature must also be compatible with the overload signatures. For example, these functions have errors because the implementation signature doesn’t match the overloads in a correct way:

Try this code ↗

function fn(x: boolean): void;
// Argument type isn't right
functionfn(x: string): void;
functionfn(x: boolean) {}
Generated error
This overload signature is not compatible with its implementation signature.

Try this code ↗

function fn(x: string): string;
// Return type isn't right
functionfn(x: number): boolean;
functionfn(x: string | number) {
return"oops";
}
Generated error
This overload signature is not compatible with its implementation signature.

Writing Good Overloads

Like generics, there are a few guidelines you should follow when using function overloads. Following these principles will make your function easier to call, easier to understand, and easier to implement.

Let’s consider a function that returns the length of a string or an array:

Try this code ↗

function len(s: string): number;
functionlen(arr: any[]): number;
functionlen(x: any) {
returnx.length;
}

This function is fine; we can invoke it with strings or arrays. However, we can’t invoke it with a value that might be a string or an array, because TypeScript can only resolve a function call to a single overload:

Try this code ↗

len(""); // OK
len([0]); // OK
len(Math.random() > 0.5 ? "hello" : [0]);
Generated error
No overload matches this call.
  Overload 1 of 2, '(s: string): number', gave the following error.
    Argument of type 'number[] | "hello"' is not assignable to parameter of type 'string'.
      Type 'number[]' is not assignable to type 'string'.
  Overload 2 of 2, '(arr: any[]): number', gave the following error.
    Argument of type 'number[] | "hello"' is not assignable to parameter of type 'any[]'.
      Type 'string' is not assignable to type 'any[]'.

Because both overloads have the same argument count and same return type, we can instead write a non-overloaded version of the function:

Try this code ↗

function len(x: any[] | string) {
returnx.length;
}

This is much better! Callers can invoke this with either sort of value, and as an added bonus, we don’t have to figure out a correct implementation signature.

Always prefer parameters with union types instead of overloads when possible

Declaring this in a Function

TypeScript will infer what the this should be in a function via code flow analysis, for example in the following:

Try this code ↗

const user = {
id:123,

admin:false,
becomeAdmin:function () {
this.admin = true;
  },
};

TypeScript understands that the function user.becomeAdmin has a corresponding this which is the outer object user. this, heh, can be enough for a lot of cases, but there are a lot of cases where you need more control over what object this represents. The JavaScript specification states that you cannot have a parameter called this, and so TypeScript uses that syntax space to let you declare the type for this in the function body.

Try this code ↗

interface DB {
filterUsers(filter: (this: User) =>boolean): User[];
}

constdb = getDB();
constadmins = db.filterUsers(function (this: User) {
returnthis.admin;
});

This pattern is common with callback-style APIs, where another object typically controls when your function is called. Note that you need to use function and not arrow functions to get this behavior:

Try this code ↗

interface DB {
filterUsers(filter: (this: User) =>boolean): User[];
}

constdb = getDB();
constadmins = db.filterUsers(() =>this.admin);
Generated error
The containing arrow function captures the global value of 'this'.Element implicitly has an 'any' type because type 'typeof globalThis' has no index signature.

Other Types to Know About

There are some additional types you’ll want to recognize that appear often when working with function types. Like all types, you can use them everywhere, but these are especially relevant in the context of functions.

void

void represents the return value of functions which don’t return a value. It’s the inferred type any time a function doesn’t have any return statements, or doesn’t return any explicit value from those return statements:

Try this code ↗

// The inferred return type is void
functionnoop() {
return;
}

In JavaScript, a function that doesn’t return any value will implicitly return the value undefined. However, void and undefined are not the same thing in TypeScript. There are further details at the end of this chapter.

void is not the same as undefined.

object

The special type object refers to any value that isn’t a primitive (string, number, bigint, boolean, symbol, null, or undefined). This is different from the empty object type{ }, and also different from the global type Object. It’s very likely you will never use Object.

object is not Object. Always use object!

Note that in JavaScript, function values are objects: They have properties, have Object.prototype in their prototype chain, are instanceof Object, you can call Object.keys on them, and so on. For this reason, function types are considered to be objects in TypeScript.

unknown

The unknown type represents any value. This is similar to the any type, but is safer because it’s not legal to do anything with an unknown value:

Try this code ↗

function f1(a: any) {
a.b(); // OK
}
functionf2(a: unknown) {
a.b();
}
Generated error
'a' is of type 'unknown'.

This is useful when describing function types because you can describe functions that accept any value without having any values in your function body.

Conversely, you can describe a function that returns a value of unknown type:

Try this code ↗

function safeParse(s: string): unknown {
returnJSON.parse(s);
}

// Need to be careful with 'obj'!
constobj = safeParse(someRandomString);

never

Some functions never return a value:

Try this code ↗

function fail(msg: string): never {
thrownewError(msg);
}

The never type represents values which are never observed. In a return type, this means that the function throws an exception or terminates execution of the program.

never also appears when TypeScript determines there’s nothing left in a union.

Try this code ↗

function fn(x: string | number) {
if (typeofx === "string") {
// do something
  } elseif (typeofx === "number") {
// do something else
  } else {
x; // has type 'never'!
  }
}

Function

The global type Function describes properties like bind, call, apply, and others present on all function values in JavaScript. It also has the special property that values of type Function can always be called; these calls return any:

Try this code ↗

function doSomething(f: Function) {
returnf(1, 2, 3);
}

This is an untyped function call and is generally best avoided because of the unsafe any return type.

If you need to accept an arbitrary function but don’t intend to call it, the type () => void is generally safer.

Rest Parameters and Arguments

Background Reading:Rest Parameters ↗Spread Syntax ↗

Rest Parameters

In addition to using optional parameters or overloads to make functions that can accept a variety of fixed argument counts, we can also define functions that take an unbounded number of arguments using rest parameters.

A rest parameter appears after all other parameters, and uses the ... syntax:

Try this code ↗

function multiply(n: number, ...m: number[]) {
returnm.map((x) =>n * x);
}
// 'a' gets value [10, 20, 30, 40]
consta = multiply(10, 1, 2, 3, 4);

In TypeScript, the type annotation on these parameters is implicitly any[] instead of any, and any type annotation given must be of the form Array<T> or T[], or a tuple type (which we’ll learn about later).

Rest Arguments

Conversely, we can provide a variable number of arguments from an iterable object (for example, an array) using the spread syntax. For example, the push method of arrays takes any number of arguments:

Try this code ↗

const arr1 = [1, 2, 3];
constarr2 = [4, 5, 6];
arr1.push(...arr2);

Note that in general, TypeScript does not assume that arrays are immutable. This can lead to some surprising behavior:

Try this code ↗

// Inferred type is number[] -- "an array with zero or more numbers",
// not specifically two numbers
constargs = [8, 5];
constangle = Math.atan2(...args);
Generated error
A spread argument must either have a tuple type or be passed to a rest parameter.

The best fix for this situation depends a bit on your code, but in general a const context is the most straightforward solution:

Try this code ↗

// Inferred as 2-length tuple
constargs = [8, 5] asconst;
// OK
constangle = Math.atan2(...args);

Using rest arguments may require turning on downlevelIteration when targeting older runtimes.

Parameter Destructuring

Background Reading:Destructuring Assignment ↗

You can use parameter destructuring to conveniently unpack objects provided as an argument into one or more local variables in the function body. In JavaScript, it looks like this:

function sum({ a, b, c }) {
console.log(a + b + c);
}
sum({ a:10, b:3, c:9 });

The type annotation for the object goes after the destructuring syntax:

Try this code ↗

function sum({ a, b, c }: { a: number; b: number; c: number }) {
console.log(a + b + c);
}

This can look a bit verbose, but you can use a named type here as well:

Try this code ↗

// Same as prior example
typeABC = { a: number; b: number; c: number };
functionsum({ a, b, c }: ABC) {
console.log(a + b + c);
}

Assignability of Functions

Return type void

The void return type for functions can produce some unusual, but expected behavior.

Contextual typing with a return type of void does not force functions to not return something. Another way to say this is a contextual function type with a void return type (type voidFunc = () => void), when implemented, can return any other value, but it will be ignored.

Thus, the following implementations of the type () => void are valid:

Try this code ↗

type voidFunc = () => void;

constf1: voidFunc = () => {
returntrue;
};

constf2: voidFunc = () =>true;

constf3: voidFunc = function () {
returntrue;
};

And when the return value of one of these functions is assigned to another variable, it will retain the type of void:

Try this code ↗

const v1 = f1();

constv2 = f2();

constv3 = f3();

This behavior exists so that the following code is valid even though Array.prototype.push returns a number and the Array.prototype.forEach method expects a function with a return type of void.

Try this code ↗

const src = [1, 2, 3];
constdst = [0];

src.forEach((el) =>dst.push(el));

There is one other special case to be aware of, when a literal function definition has a void return type, that function must not return anything.

Try this code ↗

function f2(): void {
// @ts-expect-error
returntrue;
}

constf3 = function (): void {
// @ts-expect-error
returntrue;
};

For more on void please refer to these other documentation entries:

Last updated on