This essay is an unpublished work in progress. Please check back for updates. If you choose to share it, please mention that it is unfinished at this time.
In this essay, we’re going to work with invertible functions. A function is invertible if it has an inverse function. Consider:
const evenToNatural = even => even / 2;
const naturalToEven = natural => natural * 2;
evenToNatural
and naturalToEven
are inverses of each other:
evenToNatural(14)
//=> 7
naturalToEven(7)
//=> 14
Which is to say, each function inverts or reverses the computation of the other function. This implies that for all n ∈ N
:
n === evenToNatural(naturalToEven(n)) &&
n === naturalToEven(evenToNatural(n))
And, subject to JavaScript’s implementation limitations, this expression is true
.1
A function f
is considered invertible if there also exists a function g
such that f
and g
are inverses of each other. evenToNatural
is an invertible function, because naturalToEven
is its inverse, and vice-versa.
The following functions are also invertible, even though we do not give their inverses:
const increment = n => n + 1;
const cons = ([head, tail]) => [head, ...tail];
The cons
function given implies that where compound types like lists or maps are concerned, we base our reasoning on structural equality, not referential equality.2
Some functions are their own inverse. This still makes them invertible functions. I
—also known as the “identity” or “idiot bird” function—is its own inverse.3
Other functions that are their own inverses include:
const oddsAndEvens = n => n % 2 === 0 ? n + 1 : n - 1;
const positiveNegative = n => -n;
functions that return the same output for different inputs aren’t invertible
For a function to be invertible, it must map exactly one elements of its input to exactly one element of its output. Thus, any function for which two or more different inputs produce the same output cannot be invertible:
const square = n => n * n;
const roundTen = Math.floor(n / 10) * 10;
const collatz = n => n % 2 === 0 ? n / 2 : 3 * x + 1;
const butFirst = ([first, ...rest]) => rest;
square
is not invertible because every positive output has two possible inputs, one positive and one negative. Presuming that we are sticking with natural numbers, roundTen
has ten possible inputs for each output. And collatz
has multiple possible inputs as well: For example, if the output is 10
, the input could be 3
or 20
.4
When there are multiple inputs for the same output, what would an inverse of the function return for that output? For example, if we assume the function ztalloc
inverts collatz
, what does ztalloc(10)
return? Three? Or Twenty?
Finally, butFirst
isn’t invertible because it has infinitely many inputs that produce the same output. For example, all lists of length one produce as output the empty list []
.
Returning two different values where one is expected isn’t possible, which is why a function that is invertible must have exactly one unique input for each unique output.
maps can be invertible
Consider the maps lettersToNumbers
and numbersToLetters
:
const az = {
a: 0, b: 1, c: 2, d: 3,
e: 4, f: 5, g: 6, h: 7,
i: 8, j: 9, : 10, l: 11,
m: 12, n: 13, o: 14, p: 15,
q: 16, r: 17, s: 18, t: 19,
u: 20, v: 21, w: 22, x: 23,
y: 24, z: 25
};
const lettersToNumbers = new Map(Object.entries(az));
const numbersToLetters = new Map(Object.entries(az).map(([k, v]) => [v, k]));
lettersToNumbers.get('c')
//=> 2
numbersToLetters.get(2)
//=> 'c'
Because the original az
POJO maps a unique set of values to another unique set of values, lettersToNumbers
and numbersToLetters
are inversions of each other. They aren’t invertible functions, because in JavaScript a Map is not a function. In some other languages–like Clojure–maps are automatically functions.
But this being JavaScript, we can write some code to convert our maps into invertible functions:
const numberOf = letter => lettersToNumbers.get(letter);
const letterOf = number => numbersToLetters.get(number);
numberOf('c')
//=> 2
letterOf(2)
//=> 'c'
Note that by convention, undefined
is not considered a value when we work with invertible functions. Thus, the following does not mean that our functions return the same value for two different inputs:
numberOf(6)
//=> undefined
numberOf(42)
//=> undefined
letterOf('snafu')
//=> undefined
letterOf('fubar')
//=> undefined
It literally just means that our functions do not define an output for those inputs.
trivially invertible functions
A function that only produces one output—a constant function—can be invertible, provided it only accepts one input. Such functions are called trivially invertible. This follows directly from our invertible maps. What if an invertible map only maps one key to one value?
const m = {
here: 'there'
};
const fromHereToThere = new Map(Object.entries(m));
const fromThereToHere = new Map(Object.entries(m).map(([k, v]) => [v, k]));
const toThere = from => fromHereToThere.get(from);
const toFrom = there => fromThereToHere.get(there);
toThere('here')
//=> 'there'
toFrom('there')
//=> 'here'
invertible functions and compound values
Invertible functions can’t have two outputs. They also can’t have two inputs. Therefore, this function is not invertible as written:
const cons = (head, tail) => [head, ...tail];
Instead, we must find a way to represent multiple inputs as a single compound value.
For example, we can rewrite our function to take a single tuple as input, and rely on the semantics of structural equality, as we mentioned above:2
const cons = ([head, tail]) => [head, ...tail];
Now our function can be inverted:
const carcdr = ([head, ...tail]) => [head, tail];
In general, all functions mapping multiple inputs to one output can be rewritten as taking a list or other compound value as an input and returning a single value.
Composing Invertible Functions
If f
and g
are invertible functions, and ⁻¹
as a suffix denotes “the inverse of” a function, then (f ֯ g)⁻¹ = (g⁻¹ ֯ f⁻¹)
.
Thus, if we compose two invertible functions, the inverse of that composition can be computed by taking the composition of the inverses of each function, in reverse order:
const plusOne = n => n + 1;
const minusOne = n => n - 1;
const timesTwo = n => n * 2;
const dividedByTwo = n => n / 2;
const compose = fns => argument =>
fns.reduceRight((value, fn) => fn(value), argument);
const plusOneTimesTwo = compose([plusOne, timesTwo]);
const dividedByTwoMinusOne = compose([dividedByTwo, minusOne]);
plusOneTimesTwo(42)
//=> 85
dividedByTwoMinusOne(85 )
//=> 42
Keeping track of functions and their inverses can be a bit of a headache, especially as we compose invertible functions. To keep things relatively simple, we’ll introduce an object to do the work for us:
const R = {
_inverses: new Map(),
add(f, g) {
this._inverses.set(f, g);
this._inverses.set(g, f);
return f;
},
has(f) {
return this._inverses.has(f);
},
get(f) {
return this._inverses.get(f);
}
};
R.add(plusOneTimesTwo, dividedByTwoMinusOne);
R.get(dividedByTwoMinusOne)(42)
//=> 85
R.get(plusOneTimesTwo)(85)
//=> 42
Let’s write a compose
method for R
that composes a list of invertible functions and their inverses simultaneously:
const R = {
// ...
compose(fns) {
const composition = argument =>
fns.reduceRight((value, fn) => fn(value), argument);
const inverse = argument =>
fns.reduce((value, fn) => this.get(fn)(value), argument);
return this.add(composition, inverse);
}
};
R.add(plusOne, minusOne);
R.add(timesTwo, dividedByTwo);
plusOneTimesTwo(42)
//=> 85
R.get(plusOneTimesTwo)(85)
//=> 42
We’ll use this later.
converting between numbers and their binary representation
Here’s a function that converts a positive natural number into the list form of its binary representation, without relying on JavaScript’s capabilities for parsing and representing numbers in various bases:
const toBinaryNaive = n => {
const b = [];
while (n !== 0) {
const bit = n % 2;
n = Math.floor(n / 2);
b.unshift(bit);
}
return b;
};
toBinaryNaive(1)
//=> [1]
toBinaryNaive(6)
//=> [1, 1, 0]
toBinaryNaive(23)
//=> [1, 0, 1, 1, 1]
And here’s its inverse, a function that converts the list form of a positive natural number’s binary representation into the number:
function fromBinaryNaive(b) {
let n = 0;
for (const bit of b) {
n *= 2;
n += bit;
}
return n;
};
fromBinaryNaive([1])
//=> 1
fromBinaryNaive([1, 1, 0])
//=> 6
fromBinaryNaive([1, 0, 1, 1, 1])
//=> 23
We can tell from careful inspection that for positive naturals within implementation bounds, toBinaryNaive
and fromBinaryNaive
are inverses of each other. But even with such a simple function, it requires examination to determine that they are inverses of each other.
This is especially true because the two functions are written in different styles, one uses a while
loop, the other a for... of
loop, and the ways in which they do basic arithmetic aren’t obviously symmetrical the way n => n + 1
and n => n - 1
are.
When we used simple composition of invertible functions, we were able to derive the composition of their inverses automatically. It’s possible to use that same approach to make more complex functions like toBinary
. We’ll see how in the next section.
refactoring to use invertible functions
We’re going to refactor these two invertible functions, beginning by extracting a pair of inverses at the core of each function’s loop:
const divideByTwoWithRemainder = R.add(
n => [Math.floor(n / 2), n % 2],
([n, r]) => n * 2 + r
);
Now we can write:
const toBinaryRefactored = n => {
const b = [];
let bit;
while (n > 0) {
[n, bit] = divideByTwoWithRemainder(n);
b.unshift(bit);
}
return b;
};
toBinaryRefactored(1)
//=> [1]
toBinaryRefactored(6)
//=> [1, 1, 0]
toBinaryRefactored(23)
//=> [1, 0, 1, 1, 1]
function fromBinaryRefactored(b) {
let n = 0;
for (const bit of b) {
n = R.get(divideByTwoWithRemainder)([n, bit]);
}
return n;
};
fromBinaryRefactored([1])
//=> 1
fromBinaryRefactored([1, 1, 0])
//=> 6
fromBinaryRefactored([1, 0, 1, 1, 1])
//=> 23
We’ll refactor the loops next.
refactoring to folds and unfolds
In functional programming,
fold
(also termedreduce
,accumulate
,aggregate
,compress
, orinject
) refers to a family of higher-order functions that analyze a recursive data structure and through use of a given combining operation, recombine the results of recursively processing its constituent parts, building up a return value.Fold is in a sense dual to
unfold
, which takes a seed value and apply a function corecursively to decide how to progressively construct a corecursive data structure.–Wikipedia
Here is a higher-order fold
that takes a base
(sometimes called a seed
) value, plus a combiner
function. It is specific to lists. To make it useful for our purposes, instead of the combining function taking two arguments, this fold expects its combiner function to take a one argument, a list with two elements.
It is implemented as a loop, because loops are trivially equivalent to linear recursion. It is also implemented as a higher-order function that returns a function:
const foldList = (base, combiner) =>
list => {
let folded = base;
for (const element of list) {
folded = combiner([folded, element]);
}
return folded;
};
We use it with our multiplyByTwoWithRemainder
function to generate fromBinaryFold
:
const fromBinaryFold = foldList(0, multiplyByTwoWithRemainder);
And here’s unfoldToList
. Our unfold is also written as a loop. We will use a simple equality test that works for primitive values and strings, but not objects and especially not maps of any kind.5
This unfold is written as an “eager right unfold,” which is to say, it assembles its list in the reverse order of our fold. That’s different from how unfolds are usually written, but then again, most people aren’t trying to invert a fold.
We’ll use our unfold to derive toBinaryUnfold
:
const deepEqual = (a, b) => {
if (a instanceof Array && b instanceof Array) {
if (a.length !== b.length) {
return false;
} else {
for (let i = 0; i < a.length; ++i) {
if (!deepEqual(a[i], b[i])) {
return false;
}
}
return true;
}
} else if (a instanceof Array || b instanceof Array) {
return false;
} else {
return a === b;
}
};
const unfoldToList = (base, uncombiner) =>
folded => {
let list = [];
let element;
while (!deepEqual(folded, base)) {
[folded, element] = uncombiner(folded);
list.unshift(element);
}
return list;
};
const toBinaryUnfold = unfoldToList(0, R.get(multiplyByTwoWithRemainder));
We have demonstrated another form of composition of invertible functions: Given a common base
, foldList
of an invertible function is the inverse of unfoldToList
of its inverse.
Let’s add this to R
:
const R = {
// ...
foldList(base, combiner) {
const folder = foldList(base, combiner);
const unfolder = unfoldToList(base, this.get(combiner));
return this.add(folder, unfolder);
},
unfoldToList(base, uncombiner) {
const unfolder = unfoldToList(base, uncombiner);
const folder = foldList(base, this.get(uncombiner));
return this.add(unfolder, folder);
}
};
const toBinary = R.unfoldToList(0, divideByTwoWithRemainder);
toBinary(1)
//=> [1]
toBinary(6)
//=> [1, 1, 0]
toBinary(23)
//=> [1, 0, 1, 1, 1]
const fromBinary = R.get(toBinary);
fromBinary([0])
//=> 0
fromBinary([1])
//=> 1
fromBinary([1, 1, 0])
//=> 6
fromBinary([1, 0, 1, 1, 1])
//=> 23
why refactoring to fold and unfold matters
R.foldList
, and R.unfoldToList
are higher-order invertible functions. This helps us compose new invertible functions by applying linear recursion to invertible functions like divideByTwoWithRemainder
. Once we’re satisfied that one-liners like n => [Math.floor(n / 2), n % 2]
and ([n, r]) => n * 2 + r
are invertible, we can compose toBinary
and from Binary
out of them, confident that toBinary
and from Binary
will also be invertible.
By writing higher-order functions for composing invertible functions that preserve “invertibility,” we make it easier to reason about what our code does. And so it goes for all composition: Composing functions with well-understood patterns like folding and unfolding makes it easier for us to reason about what our code does.
Now let’s look at a “contrived” problem we will solve with invertible functions.
The Night Clerk at Hilbert’s Hotel
Hilbert’s Paradox of the Grand Hotel is a thought experiment which illustrates a counterintuitive property of countably infinite sets: It demonstrates that although it may seem “obvious” that some infinite sets are larger than others, many of them are in fact equal in cardinality to the set of natural numbers.
The proposition is that the Grand Hotel has a countably infinite number of rooms, each of which is denoted by a natural number. The hotel’s Night Clerk has the problem of coming up with an algorithm for assigning an infinite number of guests to the countably infinite number of rooms.
kurt gödel and the irreducible fractions club
A positive irreducible fraction is a rational number where the numerator and denominator are coprime.6 1/1
, 1/3
, and 5/3
are irreducible fractions. 4/2
is not an irreducible fraction, because four and two are both divisible by two, and thus it can be reduced to 2/1
.
One night at the Grand Hotel, an infinite number of guests show up. They’re members of the Irreducible Fractions Club, and each member of the club is given their own unique irreducible fraction as an identifier.
The Night Clerk has to assign the guests to rooms, and comes up with an idea: Write an invertible function that maps positive irreducible numbers to positive natural numbers. That way, each guest can use the function mapping positive irreducible fractions to natural numbers to find their room. And given an occupied room, we can use the function mapping positive natural numbers to positive irreducible fractions to find the guest.
In JavaScript terms, we need a function that maps lists of two coprime numbers to numbers, and another that maps numbers to lists of two coprime numbers. The Night Clerk’s first attempt is based on prime factorization:7
const guestToRoom = ([n, d]) => 2**n * 3**d;
const roomToGuest = n => [2, 3].reduce(
(acc, prime) => {
let exponent = 0;
while (n >= prime && n % prime === 0) {
++exponent;
n /= prime;
}
return acc.concat([exponent]);
},
[]
);
guestToRoom([1, 1])
//=> 6
guestToRoom([1, 2])
//=> 18
guestToRoom([3, 2])
//=> 72
guestToRoom([1, 3])
//=> 54
guestToRoom([2, 1])
//=> 12
guestToRoom([3, 1])
//=> 24
guestToRoom([2, 3])
//=> 108
This works fine, but it assigns guests to a subset of the rooms of the Grand Hotel. It’s undesirable to have empty rooms at any hotel, even one with an infinite number of rooms. And from a maths perspective, this doesn’t demonstrate that there are an equal number of positive irreducible fractions as there are positive natural numbers, because there are an infinite number of positive natural numbers that are not associated with a positive irreducible fraction, e.g. the numbers one through five, and anything divisible by a prime larger than three.
The Night Clerk decides to try again, this time taking advantage of an interesting rule about the relationship between members of the irreducible fraction club.
the irreducible fractions club family tree
The Irreducible Fractions Club has an interesting rule: Each member of the club is required to recruit exactly two more members of the club, who in turn must recruit two more members of the club, and so forth. Furthermore, they have a specific rule for assigning irreducible fractions to new members.
The founder’s fraction was 1 / 1
. The founder’s recruits were the fractions 2 / 1
and 1 / 2
. Their recruits were given the fractions 3 / 1
, 2 / 3
and 3 / 2
, 1 / 3
respectively. And this tree of recruits and names continues, to infinity. The rules for constructing the Irreducible Fractions Club’s “family tree” are as follows:
- All members are given a unique irreducible fraction;
- Irreducible fractions are of the form
n / d
, representing a numerator and denominator; - Every member has two recruits. If the member is
n / d
, their two recruits are alwaysn + d / d
andn / n + d
. - The founder is
1 / 1
.
From this, the tree grows “upwards” to infinity:
The club claims that every irreducible fraction appears exactly once somewhere in the tree, and therefore, there is exactly one club member for every irreducible fraction.8
The Night Clerk observes that if every irreducible fraction appears exactly once somewhere in the tree, there is a unique path from the founding fraction 1 / 1
to every irreducible fraction, and if we label the arcs from each parent to child with a 0
and a 1
, each unique path has a unique encoding as a list of 0
s and 1
s.
For example, the path from 1 / 1
to 2 / 5
encoded as [1, 1, 0]
, while the path from 1 / 1
to 1 / 2
is encoded as [0, 0]
:
mapping irreducible fractions to paths
Let’s formalize an algorithm for going from irreducible fractions to paths. Given some irreducible fraction n / d
, we know that if n
and d
are both 1
, the path will be []
(the empty path). If it’s some other irreducible fraction, we have to find its parent and determine whether the arc between fraction and parent is labeled 0
or 1
.
At each step, we go from n / d
to either n / d - n
or n - d / d
. And if n < d
, we go from n / d
to n / d - n
along a path labeled 0
. While if n > d
, we go from n / d
to n - d / d
along a path labeled 1
.
That’s enough to write an unfold, representing irreducible fractions as a list with two numbers:
const irreducibleToParentAndArc = ([n, d]) =>
n < d ? [[n, d - n], 0] : [[n - d, d], 1];
const irreducibleToPath = unfoldToList([1, 1], irreducibleToParentAndArc);
irreducibleToPath([2, 5])
//=> [1, 0, 0]
irreducibleToPath([1, 3 ])
//=> [0, 0]
Hmmm. If we can write an unfold around the uncombiner function ([n, d]) => n < d ? [[n, d - n], 0] : [[n - d, d], 1]
, does that mean that if we can write an inverse, we can get a fold as well? Yes:
const parentAndArcToIrreducible = ([[n, d], arc]) =>
arc === 0 ? [n, n + d] : [n + d, d];
const toIrreducible = foldList([1, 1], parentAndArcToIrreducible);
toIrreducible([1, 0, 0])
//=> [2, 5]
toIrreducible([0, 0])
//=> [1, 3]
And from this, it follows that:
const toIrreducible = R.foldList(
[1, 1],
R.add(
([[n, d], arc]) =>
arc === 0 ? [n, n + d] : [n + d, d],
([n, d]) =>
n < d ? [[n, d - n], 0] : [[n - d, d], 1]
)
);
toIrreducible([1, 0, 0])
//=> [2, 5]
toIrreducible([0, 0])
//=> [1, 3]
R.get(toIrreducible)([2, 5])
//=> [1, 0, 0]
R.get(toIrreducible)([1, 3])
//=> [0, 0]
a wild infinite loop appears
The Night Clerk has one more step to complete: Going from positive natural numbers to paths. We have already seen invertible functions for converting between positive natural numbers and binary representations, but there’s a problem. Let’s take a look:
1 : [1]
2 : [1, 0]
3 : [1, 1]
4 : [1, 0, 0]
5 : [1, 0, 1]
6 : [1, 1, 0]
7 : [1, 1, 1]
8 : [1, 0, 0, 0]
9 : [1, 0, 0, 1]
10 : [1, 0, 1, 0]
11 : [1, 0, 1, 1]
12 : [1, 1, 0, 0]
13 : [1, 1, 0, 1]
14 : [1, 1, 1, 0]
15 : [1, 1, 1, 1]
16 : [1, 0, 0, 0, 0]
17 : [1, 0, 0, 0, 1]
18 : [1, 0, 0, 1, 0]
19 : [1, 0, 0, 1, 1]
Every number greater than zero has a binary representation that starts with a 1
. If we map these directly to paths, we only get half of the tree:
Although the binary representation cannot be taken directly as paths, it can be used. If we remove the frontmost 1
from every representation, we get an enumeration of every path in the tree:
1 : [1] : [1]
2 : [1, 0] : [0]
3 : [1, 1] : [1]
4 : [1, 0, 0] : [0, 0]
5 : [1, 0, 1] : [0, 1]
6 : [1, 1, 0] : [1, 0]
7 : [1, 1, 1] : [1, 1]
8 : [1, 0, 0, 0] : [0, 0, 0]
9 : [1, 0, 0, 1] : [0, 0, 1]
10 : [1, 0, 1, 0] : [0, 1, 0]
11 : [1, 0, 1, 1] : [0, 1, 1]
12 : [1, 1, 0, 0] : [1, 0, 0]
13 : [1, 1, 0, 1] : [1, 0, 1]
14 : [1, 1, 1, 0] : [1, 1, 0]
15 : [1, 1, 1, 1] : [1, 1, 1]
16 : [1, 0, 0, 0, 0] : [0, 0, 0, 0]
17 : [1, 0, 0, 0, 1] : [0, 0, 0, 1]
18 : [1, 0, 0, 1, 0] : [0, 0, 1, 0]
19 : [1, 0, 0, 1, 1] : [0, 0, 1, 1]
We’ll need an invertible way to remove the 1
from the head of each list. There is a way, but we must be careful. This won’t work, even though it looks like it works:
const corrigansTail = R.add(
[head, ...tail] => tail,
tail => [1, ...tail]
);
corrigansTail([1, 0, 0])
//=> [0, 0]
R.get(corrigansTail)([0, 0])
//=> [1, 0, 0]
This is the problem:
corrigansTail([7, 0, 0])
//=> [0, 0]
R.get(corrigansTail)([0, 0])
//=> [1, 0, 0]
There are inputs for which corrigansTail
will accept and output a value, but it’s proposed inverse will not take the resulting output and return the original input.
As we saw earlier when discussing invertible functions, [head, ...tail] => tail
is not invertible. It can’t be, as it loses information. It outputs fewer bits than it takes as an argument.
But consider this:
const butLastAndLast = list => [
list.slice(0, list.length - 1),
list[list.length - 1]
];
const append =
([list, element]) => [...list, element];
const tail = R.unfoldToList(
[1],
R.add(butLastAndLast, append)
);
tail([1, 0, 0])
//=> [0, 0]
R.get(tail)([0, 0])
//=> [1, 0, 0]
tail
has the same effect as corrigansTail
for lists beginning with 1
, but when we give it a list beginning with another number, it does not return a result:
tail([7, 0, 0])
//=> 💀
Functions that don’t return are undesirable in production, but from a semantic perspective, they fit the bill as much as functions that raise helpful exceptions. To be invertible, there must exist a function such that for every value that ttail
returns, its inverse takes tail
’s output as its input and returns tail
’s input.
Input that don’t produce outputs don’t count, so tail
is invertible: Every output it actually produces is an input to its inverse function.
Finding Rooms for The Irreducible Fractions Club
The Night Clerk now has what they need to map positive invertible fractions to positive natural numbers, which means that given a member of the Irreducible Fractions Club, they can assign them to a room in the Grand Hotel:
const binaryToNumber = R.foldList(
0,
R.add(
([n, r]) => n * 2 + r,
n => [Math.floor(n / 2), n % 2]
)
);
const pathToBinary = R.foldList(
[1],
R.add(
([list, element]) =>
[...list, element],
list =>
[list.slice(0, list.length - 1), list[list.length - 1]]
)
);
const irreducibleToPath = R.unfoldToList(
[1, 1],
R.add(
([n, d]) =>
n < d ? [[n, d - n], 0] : [[n - d, d], 1],
([[n, d], arc]) =>
arc === 0 ? [n, n + d] : [n + d, d]
)
);
const roomNumber = R.compose([
binaryToNumber,
pathToBinary,
irreducibleToPath
]);
roomNumber([1,1])
//=> 1
roomNumber([1,2])
//=> 2
roomNumber([2,1])
//=> 3
roomNumber([1,3])
//=> 4
roomNumber([3,2])
//=> 5
roomNumber([2,3])
//=> 6
roomNumber([3,1])
//=> 7
roomNumber([1,4])
//=> 8
roomNumber([4,3])
//=> 9
roomNumber([3,5])
//=> 10
roomNumber([5,2])
//=> 11
roomNumber([2,5])
//=> 12
roomNumber([5,3])
//=> 13
roomNumber([3,4])
//=> 14
roomNumber([4,1])
//=> 15
The roomNumber
function is composed out of invertible functions, and uses composition that preserves invertibility, so of course it is invertible, giving us a function for computing the occupant, given a room number:
const occupant = R.get(roomNumber);
occupant(1)
//=> [1,1]
occupant(2)
//=> [1,2]
occupant(3)
//=> [2,1]
occupant(4)
//=> [1,3]
occupant(5)
//=> [3,2]
occupant(6)
//=> [2,3]
occupant(7)
//=> [3,1]
occupant(8)
//=> [1,4]
occupant(9)
//=> [4,3]
occupant(10)
//=> [3,5]
occupant(11)
//=> [5,2]
occupant(12)
//=> [2,5]
occupant(13)
//=> [5,3]
occupant(14)
//=> [3,4]
occupant(15)
//=> [4,1]
The Night Clerk can now say with confidence that every room in the Grand Hotel has been occupied by a member of the Irreducible Fractions Club, and that every member of the Irreducible Fractions Club occupies a room. This establishes that there are exactly as many positive irreducible fractions as there are positive natural numbers.
what can we learn from this?
The need for composing invertible functions comes up rarely, if at all. However, we often compose functions. And from our examination of invertible functions, we see that one value of composing functions with standard composition tools is that they make it easier for us to reason about what our code does.
This is no different than the the innovation of what was one called “Structured Programming.” By limiting looping code to specific constructs (e.g. for
, while
, and do... while
), programmers discovered they had just as much power as when they used GOTOs and registers, but the code was easier to understand.
A generation later, programmers discovered that constructs like mapping, reducing, and transducers again provided the same power as loops, but were easier to reason about. And so it is with composing functions: Carefully chosen constructs can provide the same power, but also make it easier to reason about the code.
Our takeaway is thus:
- Prefer composition to just about everything else;
- When composing, prefer patterns that make it easier to reason about the functions we’re composing.
Concatenative Invertibility, Wherein We Make a Language
Purely invertible functions may be rare in everyday programming, but many programming languages–including JavaScript–provide support for destructuring assignment with built-in compound data types like POJOs and Arrays. In this section, we’ll explore how invertible functions can be used to implement destructuring assignment.
Before we get into the main work, we’ll begin with a new kind of invertible function, the invertible generator. Then we’ll talk about stackified invertible generators, invertible functions that map from a stack to a stack.
invertible generators
Because functions only return one value in fundamental computation models like the lambda calculus, and because functions in languages like JavaScript cannot return more than one value, an invertible function can only accept one value as an argument and only return one value.
But there is a way to return multiple values from a function in JavaScript, and that way is to make the function return a generator. Here’re cons
and carcdr
implemented as simple generators:
function* cons(head, tail) {
yield [head, ...tail];
}
function* carcdr([head, ...tail]) {
yield head;
yield tail;
}
function log(expr) {
for (const value of expr]) {
console.log(value)
}
}
log(carcdr(...cons(1, [2, 3]))
//=> [1, 2, 3]
log(carcdr([1, 2, 3]))
//=> 1
//=> [2, 3]
Liberated from the constraints of taking exactly one argument and returning exactly one non-undefined value, our invertible generators can take zero or more arguments, and yield zero or more values.
stackification
Now let’s consider stackifying invertible generators. stackify
is a function that takes an invertible generator (a generator function from that takes zero or more inputs and yields zero or more outputs) as an argument, and returns a function that takes a stack as an argument and returns a stack.
const stackify = fn => stack => {
if (stack.length < fn.length) return;
const inputs = stack.slice(stack.length - fn.length);
const remainder = stack.slice(0, stack.length - fn.length);
const outputs = [...fn(...inputs)];
if (outputs.every(o => o !== undefined)) {
return remainder.concat([...fn(...inputs)])
}
}
const stackifiedCons = stackify(cons);
const stackifiedCarCdr = stackify(carcdr);
stackifiedCons([1, 2, 3, 4, [5, 6]])
//=> [1, 2, 3, [4, 5, 6]]
stackifiedCarCdr([1, 2, 3, [4, 5, 6]])
//=> [1, 2, 3, 4, [5, 6]]
A stackified invertible generator becomes an ordinary invertible function that instead of mapping zero or more inputs to zero or more outputs, maps a stack to a stack. It’s just that it pops the arguments it needs from the top of the stack, and then pushes the values it return back onto the top. (Well, actually, it makes a copy of the original stack, but this is semantically what it does to the copy.)
The interesting things about “stackification” is this: If some function fn
is invertible, then stackify(fn)
is also invertible. Stackification is a way of taking functions that take zero or more arguments and return zero or more values, and turning them into properly invertible functions that take exactly one argument and return exactly one value.
We’ll see later that they have another important property, but first let’s add stackify
to R
:
const R = {
// ...
stackify(invertibleGenerator) {
const stackify = fn => stack => {
if (stack.length < fn.length) return;
const inputs = stack.slice(stack.length - fn.length);
const remainder = stack.slice(0, stack.length - fn.length);
const outputs = [...fn(...inputs)];
if (outputs.every(o => o !== undefined)) {
return remainder.concat(outputs);
}
};
return this.add(
stackify(invertibleGenerator),
stackify(this.get(invertibleGenerator))
);
}
};
const cons = R.stackify(
R.add(
function*(head, tail) {
yield [head, ...tail];
},
function*([head, ...tail]) {
yield head;
yield tail;
}
)
);
cons([1, 2, 3, []])
//=> [1, 2, [3]]
R.get(cons)([1, 2, [3]])
//=> [1, 2, 3, []]
push and pop
Consider this invertible generator:
const one = R.stackify(
R.add(
function*() => { yield 1; },
function*(n) => {
if (n !== 1) yield undefined;
}
)
);
one([5, 4, 3, 2])
//=> [5, 4, 3, 2, 1]
R.get(one)([5, 4, 3, 2, 1])
//=> [5, 4, 3, 2]
R.get(one)([5, 4, 3, 2, 1, 0])
//=> undefined
We have found another way to write an invertible function that appends or removes a specific value from a list. We can generalize this idea and add it to R
:
const R = {
// ...
push(value) {
return R.stackify(
R.add(
function*() { yield value; },
function*(input) {
if (input !== value) yield undefined;
}
)
)
}
};
const fortyTwo = R.push(42);
fortyTwo([1, 2, 3])
//=> [1, 2, 3, 42]
R.get(fortyTwo)([1, 2, 3, 42])
//=> [1, 2, 3]
Now we’re ready for the big insight.
Concatenative Programming Languages
A concatenative programming language is formally defined as, a point-free computer programming language in which all expressions denote functions, and the juxtaposition of expressions denotes function composition.
Taking that step-by step:
- There are no bindings of values to labels via arguments, functions operate directly upon each other.
- Every expression written in the evaluates to a function, nothing else.
- If you write two expressions next to each other, e.g.
expr1 expr2
, you are writing two expressions that evaluate to functions (this is given by the previous rule). Which gives usfn1 fn2
. The value of this expression is the functional composition of the two functions, equivalent tox => fn2(fn1(x))
in JavaScript.
All of this can be hard to imagine in the abstract, so we’ll consider that nearly all concatenative languages are implemented as stack machines. So we’ll describe a very simple stack machine for interpreting a concatenative language, and that will demonstrate how things work.
Our stack machine has just two kinds of storage. There is the program, which is a list of functions, and a stack:
function stackMachine(functions, stack = []) {
for (const fn of functions) {
fn(stack);
}
return stack;
}
Our programs are lists of functions, each of which takes the stack as an argument, and which we constrain to only push values on top of the stack or pop values off of the stack. Here’s a very simple program: It adds two to whatever number is on top of the stack. We run it by giving the stack machine our program and a stack with our input on top, and it returns the result on the stack:
const plusTwo = stack => {
stack.push(stack.pop() + 2);
}
stackMachine(
[plusTwo],
[1]
)
//=> [3]
We aren’t actually working with a programming language in the practical sense, of course, because we don’t have a notation for writing programs, nor do we have a lexer or parser. But this is more-or-less exactly how languages like Forth or Joy operate under the hood.
two properties of concatenating programs
Here’s a program for multiplying a number by three:
const timesThree = stack => {
stack.push(stack.pop() * 3);
}
stackMachine(
[timesThree],
[3]
)
//=> [9]
What makes concatenative languages “concatenative,” is that if we want to compose two programs, we literally concatenate their representation:
stackMachine(
[plusTwo, timesThree],
[1]
)
//=> [9]
Another thing about concatenative programs is that although our first stack machine relies upon mutating a single “global variable,” the stack, we can construct an equivalent stack machine that operates with pure functions. Instead of taking a stack as an argument and mutating it in place, our functions will take a stack as an argument and return a new stack that will replace the old:
const plusTwo = ([...stack]) => {
stack.push(stack.pop() + 2);
return stack;
}
const timesThree = ([...stack]) => {
stack.push(stack.pop() * 3);
return stack;
}
const fStackMachine(functions, stack = []) {
for (const fn of functions) {
stack = fn(stack);
}
return stack;
}
fStackMachine(
[plusTwo, timesThree],
[1]
)
//=> [9]
This illustrates the first interesting property of concatenative programs: Every concatenative program can be implemented as the simple, linear composition of pure functions.
Let’s make one more refactor, from a loop to a fold:
const pureStackMachine =
(functions, stack = []) => foldList(stack, ([s, fn]) => fn(s))(functions);
That is very interesting in the context of invertible functions, because a “stackified” invertible function is exactly the same thing as one of the functions we’re using with our stack machine:
const plusTwo = R.stackify(
R.add(
function*(n) {
yield n + 2;
},
function*(n) {
yield n - 2;
}
)
);
const timesThree = R.stackify(
R.add(
function*(n) {
yield n * 3;
},
function*(n) {
yield n / 3;
}
)
);
pureStackMachine(
[plusTwo, timesThree],
[1]
)
//=> [9]
Aha! Stackified invertible functions are equivalent to the functions used in concatenative programs. And thus we can reason:
- A stackified invertible function is an invertible function, and;
- The composition of invertible functions is invertible, therefore:
- The composition of stackified invertible functions is invertible, and thus:
- A concatenative program that consists of stackified invertible functions is an invertible concatenative program.
Let’s try it!
const inversionOf = program => program.reverse().map(fn => R.get(fn));
pureStackMachine(
inversionOf([plusTwo, timesThree]),
[9]
)
//=> 1
When we make a concatenative program out of invertible functions, we are making an invertible program.
Notes
-
We’re making a number of assumptions here that are fairly obvious. For example, we are assuming that the input to
evenToNatural
andnaturalToEven
is a number, and that the number is not one for which the underlying implementation will do something unusual. This is why actual math and actual computer science is hard: Making the jump from demonstrating that something is true to proving that it is true means embracing rigour and detail, not just focusing on the pleasant “aha!” moments. ↩ -
Structural (or “semantic”) equality is when two compound objects behave identically. Referential (or “physical”) equality is when two references refer to the same entity in the language’s implementation. JavaScript uses referential equality by default, but this is not what we want for the purposes of exploring invertible functions. ↩ ↩2
-
const I = x => x;
↩ -
We touched on The Collatz Conjecture in Remembering John Conway’s FRACTRAN, a ridiculous, yet surprisingly deep language. ↩
-
For production use, deep equality must be carefully crafted to match the semantics of the entities we wish to compare. ↩
-
Two numbers are coprime if their greatest common divisor is 1. ↩
-
Prime factorization is a trick the Night Clerk read about in Remembering John Conway’s FRACTRAN, a ridiculous, yet surprisingly deep language. ↩
-
The club is correct. The club’s rules for recruiting and assigning fractions to the recruits do guarantee that there is exactly one member for every irreducible fraction. This is a direct consequence of the Euclidean algorithm for determining the greatest common divisor of two integers. ↩