This essay takes a highly informal look at two related techniques for achieving high performance when using large data structures: Structural Sharing, and Copy-on-Write Semantics. In Part I, we’ll look at the background of Structural Sharing and start making a Slice class that abstracts the concept of a slice of an array. In Part II, we’ll consider the problem of resource ownership when mutating objects.

To give us some context for exploring these techniques, we’re going to solve a very simple problem: Programming in a Lisp-like recursive style, but using JavaScript arrays. Although not the most practical use case, it’s interesting because even a small function written in Lisp style (like summing a list of integers) can create and recycle a lot of temporary objects.

Fixing that problem gives us an excuse to look at ways to use memory efficiently, minimizing the data we have to copy. Although we’re unlikely to use recursion gratuitously to sum a list of integers, the techniques we’ll use here to make it “not embarrassing” are the exact same techniques needed to make working with large data structures performant.

And now, let’s start at the beginning. The beginning of functional programming, in fact.


The IBM 704


wherein we travel back in time to the dawn of functional programming

Once upon a time, there was a programming language called Lisp, an acronym for LIS(t) P(rocessing).1 Lisp was one of the very first high-level languages, the very first implementation was written for the IBM 704 computer.2

The 704 had a 36-bit word, meaning that it was very fast to store and retrieve 36-bit values. The CPU’s instruction set featured two important macros: CAR would fetch 15 bits representing the Contents of the Address part of the Register, while CDR would fetch the Contents of the Decrement part of the Register.

In broad terms, this means that a single 36-bit word could store two separate 15-bit values and it was very fast to save and retrieve pairs of values. If you had two 15-bit values and wished to write them to the register, the CONS macro would take the values and write them to a 36-bit word.

Thus, CONS put two values together, CAR extracted one, and CDR extracted the other. Lisp’s basic data type is often said to be the list, but in actuality it was the “cons cell,” the term used to describe two 15-bit values stored in one word. The 15-bit values were used as pointers that could refer to a location in memory, so in effect, a cons cell was a little data structure with two pointers to other cons cells.

Lists were represented as linked lists of cons cells, with each cell’s head pointing to an element and the tail pointing to another cons cell.

Having these instructions be very fast was important to those early designers: They were working on one of the first high-level languages (COBOL and FORTRAN being the others), and computers in the late 1950s were extremely small and slow by today’s standards. Although the 704 used core memory, it still used vacuum tubes for its logic. Thus, the design of programming languages and algorithms was driven by what could be accomplished with limited memory and performance.

Here’s the scheme in JavaScript, using two-element arrays to represent cons cells:

const cons = (a, d) => [a, d],
      car  = ([a, d]) => a,
      cdr  = ([a, d]) => d;

We can make a list by calling cons repeatedly, and terminating it with null:

const oneToFive = cons(1, cons(2, cons(3, cons(4, cons(5, null)))));

oneToFive
  //=> [1,[2,[3,[4,[5, null]]]]]

Notice that though JavaScript displays our list as if it is composed of arrays nested within each other like Russian Dolls, in reality the arrays refer to each other with references, so [1,[2,[3,[4,[5,null]]]]] is our way to represent:

graph LR one(( ))-->|car|a["1"] one-->|cdr|two(( )) two-->|car|b["2"] two-->|cdr|three(( )) three-->|car|c["3"] three-->|cdr|four(( )) four-->|car|d["4"] four-->|cdr|five(( )) five-->|car|e["5"] five-->|cdr|null["fa:fa-ban null"];

This is a Linked List, it’s just that those early Lispers used the names car and cdr after the hardware instructions, whereas today we use words like element and next. But it works the same way: If we want the head of a list, we call car on it:

car(oneToFive)
  //=> 1

car is very fast, it simply extracts the first element of the cons cell. And what about the rest of the list? cdr does the trick:

cdr(oneToFive)
  //=> [2,[3,[4,[5, null]]]]

That’s another linked list too:

graph LR two(( ))-->|car|b["2"] two-->|cdr|three(( )) three-->|car|c["3"] three-->|cdr|four(( )) four-->|car|d["4"] four-->|cdr|five(( )) five-->|car|e["5"] five-->|cdr|null["fa:fa-ban null"];

By extracting references from cons cells, it achieves high performance. In Lisp, it’s blazingly fast because it happens in hardware. There’s no making copies of arrays, the time to get the cdr of a list with five elements is the same as the time to get the cdr pf a list with 5,000 elements. In each case, we have a reference to the first cell, and we get a reference to the next cell in one step. No elements need to be copied and the list does not need to be traversed.

In JavaScript, even without the low-level support, it’s still much, much, much faster to get all the elements except the head from a linked list than from an array. Getting one reference to a structure that already exists is faster than copying a bunch of elements.

So now we understand that in Lisp, a lot of things use linked lists, and they do that in part because it was what the hardware made fast.


Symbolics "old style" keyboard

Symbolics, Inc. was a computer manufacturer headquartered in Cambridge, Massachusetts, and later in Concord, Massachusetts, with manufacturing facilities in Chatsworth, California. Symbolics designed and manufactured a line of Lisp machines, single-user computers optimized to run the Lisp programming language.


operating on lists

As we can see, it was always fast to get the first element of a list and the rest of a list. Now, you could get every element of a list by traversing the list pointer by pointer. So if you wanted to do something with a list, like sum the elements of a list, you’d write a linearly recursive function like this:

const cons = (a, d) => [a, d],
      car  = ([a, d]) => a,
      cdr  = ([a, d]) => d;

function sum (linkedList, runningTotal = 0) {
  if (linkedList == null) {
    return runningTotal;
  } else {
    const first = car(linkedList);
    const rest = cdr(linkedList);

    return sum(rest, runningTotal + first);
  }
}

const oneToFive = cons(1, cons(2, cons(3, cons(4, cons(5, null)))));

sum(oneToFive)
  //=> 15

If we ignore the fact that the original cons cells were many many orders of magnitude faster than using arrays with two elements, we have the general idea:

It was ridiculously fast to separate a list into the first and rest, and as a result, many linear algorithms written in Lisp were organized around repeatedly (by recursion or looping) getting the first and rest of a list.


Garbage Day


Garbage, Garbage Everywhere

But what about today’s JavaScript? Today, we can write a list with an array. And we can get the first and rest with [0] and .slice(1):

function sum (array, runningTotal = 0) {
  if (array.length === 0) {
    return runningTotal;
  } else {
    const first = array[0];
    const rest = array.slice(1);

    return sum(rest, runningTotal + first);
  }
}

const oneToFive = [1, 2, 3, 4, 5];

sum(oneToFive)
  //=> 15

Like car, calling array[0] is fast. But when we invoke array.slice(1), JavaScript makes a new array that is a copy of the old array, omitting element 0. That is much slower, and since these copies are temporary, hammers away at the garbage collector.

We’re only working with five elements at a time, so we can afford to chuckle at the performance implications. But if we start operating on long lists, all that copying is going to bury us under a mound of garbage. Of course, we could switch to linked lists in JavaScript. But the cure would be worse than the disease.

Nobody wants to read code that looks like cons(1, cons(2, cons(3, cons(4, cons(5, null))))). And sometimes, we want to access arbitrary elements of a list. With a linked list, we have to traverse the list element by element to get it:3

function at (linkedList, index) {
  if (linkedList == null) {
    return undefined;
  } else if (index === 0) {
    return car(linkedList);
  } else {
    return at(cdr(linkedList), index - 1);
  }
}

const oneToFive = [1, 2, 3, 4, 5];

at(oneToFive, 4)
  //=> 5

Accessing arbitrary elements of a linked list is the “Shlemiel The Painter” of Computer Science:

Shlemiel gets a job as a street painter, painting the dotted lines down the middle of the road. On the first day he takes a can of paint out to the road and finishes 300 yards of the road. “That’s pretty good!” says his boss, “you’re a fast worker!” and pays him a kopeck.

The next day Shlemiel only gets 150 yards done. “Well, that’s not nearly as good as yesterday, but you’re still a fast worker. 150 yards is respectable,” and pays him a kopeck.

The next day Shlemiel paints 30 yards of the road. “Only 30!” shouts his boss. “That’s unacceptable! On the first day you did ten times that much work! What’s going on?” “I can’t help it,” says Shlemiel. “Every day I get farther and farther away from the paint can!”

If only there was a way to have the elegance of Lisp, and the performance of Arrays when accessing arbitrary elements.

Let’s work our way up to that. Where do we begin?


The Beginning


slicing and structural sharing

Let’s start with a couple of very modest requirements. First, what we’re building is for the case when we want to process arrays in a [0] and .slice(1), style, usually recursively.

(Most of the time, we don’t want to do process lists in this style. But when we do–perhaps we are playing with a recursive algorithm we read about in a book like SICP, perhaps we want to refactor such an algorithm step-by-step–we want the performance to be “not embarrassing.”)

Second, we are going to presume that the array we’re dealing with will not be mutated, at least not while we’re working with it. That’s certainly the case when writing functions that fold a list, like sum.

Given those two constraints, what problem are we trying to solve? As we noted, .slice(1) is expensive because it is implemented by copying arrays. Imagine an array with 10,000 elements!!! The first slice creates another array with 9,999 elements, the next with 9,998 elements, and so on.

So: Our beginning step will be to make .slice less expensive.

The technique we are going to use is called structural sharing. Let’s review our two-element array implementation of linked lists from above:

const cons = (a, d) => [a, d],
      car  = ([a, d]) => a,
      cdr  = ([a, d]) => d;

const oneToFive = cons(1, cons(2, cons(3, cons(4, cons(5, null)))));
const twoToFive = cdr(oneToFive);

The variable twoToFive points to the second element in oneToFive’s list, and both of these lists share the same four elements:

graph LR R1(oneToFive)-->one(("[...]")) R2(twoToFive)-->two(("[...]")) one-->|0|a["1"] one-->|1|two two-->|0|b["2"] two-->|1|three(("[...]")) three-->|0|c["3"] three-->|1|four(("[...]")) four-->|0|d["4"] four-->|1|five(("[...]")) five-->|0|e["5"] five-->|1|null["fa:fa-ban null"];

As long as we don’t want to destructively modify any part of a list that is being shared, this scheme works beautifully.

We are not going to use cons cells or two-element arrays, but we are going to share structure, and as noted, we are going to have to avoid any kind of operation that modifies an existing list in such a way that it affects other variables that are sharing its structure.

So what will our technique be? Well, we are going to create a data structure that behaves enough like an array that we can write things like const first = arrayLikeDataStructure[0]; and const rest = arrayLikeDataStructure.slice(1), and they will work. But of course, our implementation won’t copy arrays. Instead, it will share the array.

We’ll begin with a class representing a slice of an array. Although we don’t need them directly for our purposes, we’ll implement an iterator, a .join method, and a .toString() method, for debugging purpose:4

class Slice {
  constructor(array, from = 0, length = array.length) {
    if (from < 0) {
      from = from + array.length;
    }
    from = Math.max(from, 0);
    from = Math.min(from, array.length);

    length = Math.max(length, 0);
    length = Math.min(length, array.length - from);

    this.array = array;
    this.from = from;
    this.length = length;
  }

  * [Symbol.iterator]() {
    const { array, from, length } = this;

    for (let i = 0; i < length; i++) {
      yield array[i + from];
    }
  }

  join(separator = ",") {
    const { array, from, length } = this;

    if (length === 0) {
      return '';
    } else {
      let joined = array[from];

      for (let i = 1; i < this.length; ++i) {
        joined = joined + separator + array[from + i];
      }

      return joined;
    }
  }

  toString() {
    return this.join();
  }
}

const a1to5 = [1, 2, 3, 4, 5];
const fromTwo = new Slice(a1to5, 2);

fromTwo.toString()
  //=> "3,4,5"

[...fromTwo]
  //=> [3, 4, 5]

Instances of Slice encapsulate the idea of a slice of an array, without making another array:

graph TD b["fromTwo: { from: 2, length: 3 }"]-->|array:|a["a1to5: [1, 2, 3, 4, 5]"];

We’ll now add support for [0] and .slice(1). The function .slice is a little different from the constructor, because the constructor is concerned with initializing the object’s properties, while .slice mimics the semantics of Array.prototype.slice. And we’ll extract some duplication while we’re at it:

function normalizedFrom(arrayIsh, from = 0) {
    if (from < 0) {
      from = from + arrayIsh.length;
    }
    from = Math.max(from, 0);
    from = Math.min(from, arrayIsh.length);

    return from;
}

function normalizedLength(arrayIsh, from, length = arrayIsh.length) {
    from = normalizedFrom(arrayIsh, from);

    length = Math.max(length, 0);
    length = Math.min(length, arrayIsh.length - from);

    return length;
}

function normalizedTo(arrayIsh, from, to) {
    from = normalizedFrom(arrayIsh, from);

    to = Math.max(to, 0);
    to = Math.min(arrayIsh.length, to);

    return to;
}

class Slice {
  constructor(array, from, length) {
    this.array = array;
    this.from = normalizedFrom(array, from, length);
    this.length = normalizedLength(array, from, length);
  }

  * [Symbol.iterator]() {
    const { array, from, length } = this;

    for (let i = 0; i < length; i++) {
      yield array[i + from];
    }
  }

  join(separator = ",") {
    const { array, from, length } = this;

    if (length === 0) {
      return '';
    } else {
      let joined = array[from];

      for (let i = 1; i < this.length; ++i) {
        joined = joined + separator + array[from + i];
      }

      return joined;
    }
  }

  toString() {
    return this.join();
  }

  slice(from, to = Infinity) {
    from = normalizedFrom(this, from, length);
    to = normalizedTo(this, from, to);

    return new Slice(this.array, this.from + from, to - from);
  }
}

const a1to5 = [1, 2, 3, 4, 5];
const fromZero = new Slice(a1to5, 0);
const fromOne = fromZero.slice(1);
const twoToFour = fromZero.slice(2, 4);

[...fromOne]
  //=> [2, 3, 4, 5]
[...twoToFour]
  //=> [3, 4]
graph TD b["fromZero: { from: 0, length: 5 }"]-->|array:|a["a1to5: [1, 2, 3, 4, 5]"] c["fromOne { from: 1, length: 4 }"]-->|array:|a;

To make it work with [0], we need to implement []. Implementing [] just for 0 is easy, but if we implement just [0], we’re begging for a bug later when somebody thinks they can use [1]. What we want instead is a way to allow any indexed access, and properly access the correct element of the underlying array, and without allowing access beyond our slice’s dimension.

To do that, we’ll use a Proxy to handle indexed access.


remote-control-locomotives-sign


meta-programming with proxies

A Proxy is an object that “stands in” for another object, called the target in JavaScript’s documentation. The idea is that the proxy implements the desired behaviour of the object, so we can interact with the proxy as if it was the original.

Proxies have a number of interesting uses. One is to decorate functionality. If we wanted to log changes to a model object, one way to do that is to decorate the model’s methods with logging code. That’s the “aspect-oriented programming” approach: Add functionality to the target in a structured way.

A proxy approach would be to create a proxy for the target model, and the proxy object could implement the logging while forwarding the method invocations to the target. That separates concerns in a different way than decorating methods separates concerns. The decoration is aggregated in the proxy.

Proxies in JavaScript also provide the only way to perform dynamic method dispatch. Famously, the Ruby programming language provides a method_missing hook that allows any class to define code to handle methods that do not have concrete implementations. JavaScript does not bake that into every object. Instead, it makes this type of functionality available in proxies via specific hooks for getting and setting properties.

A proxy associates a target object with a handler object that contains—surprise—handlers for various hooks. Each hook controls a specific type of behaviour.

Initially, we’ll add a has hook and a get hook to our Slice objects. The net effect of the has hook is that every time another piece of code tries to determine whether our slice instances have a particular property, the handler intercepts the detection and can return true or false itself.

The get hook works similarly, only it is responsible for returning a value whenever another piece of code performs a property access. As a rule, it makes sense to implement these two methods in tandem.

In our case, our Slice instances do not have any properties for 0, 1, 2, &c. So if we want to be able to access the elements of the underlying array with code like someSlice[3], we need to handle the attempt to get(slice, '3') and forward it to a method we’ll write on Slice, at(...).

Of course, methods in JavaScript are functions bound to properties, so our has and get handlers always check to see if the target slice already has a property. If so, it delegates the access back to the target.

const SliceHandler = {
  has (slice, property) {
    if (property in slice) {
      return true;
    }

    if (typeof property === 'symbol') {
      return false;
    }

    const matchInt = property.match(/^\d+$/);
    if (matchInt != null) {
      const i = parseInt(property);

      return slice.has(i);
    }
  },

  get (slice, property) {
    if (property in slice) {
      return slice[property];
    }

    if (typeof property === 'symbol') {
      return;
    }

    const matchInt = property.match(/^\d+$/);
    if (matchInt != null) {
      const i = parseInt(property);
      return slice.at(i);
    }
  }
};

We also modify the Slice class. We implement has and at methods that understand the way indexes work relative to the slice’s boundaries, and we also modify the constructor to return a proxy that uses our handler to mediate access to the slice.

class Slice {
  constructor(array, from, length) {
    this.array = array;
    this.from = normalizedFrom(array, from, length);
    this.length = normalizedLength(array, from, length);

    return new Proxy(this, SliceHandler);
  }

  // ...

  has(i) {
    const { array, from, length } = this;

    if (i >= 0 && i < length) {
      return (from + i) in array;
    } else {
      return false;
    }
  }

  at(i) {
    const { array, from, length } = this;

    if (i >= 0 && i < length) {
      return array[from + i];
    }
  }
}

const a1to5 = [1, 2, 3, 4, 5];
const fromZero = new Slice(a1to5, 0);
const fromLast = new Slice(a1to5, -1);

fromZero[0]
  //=> 1
fromLast[0]
  //=> 5

In effect, our slice is a proxy (lower-case “p”) for the underlying array, and we are now returning a Proxy (upper-case “P”) for the slice. That’s two layers of proxies, and doubtless we are all thinking of the famous aphorism “All problems in computer engineering can be solved by another level of indirection, except for the problem of too many layers of indirection.”5

And now we can implement one last thing, a static factory method for making Slice objects out of other things. With of, we can to use Slice to make our recursive functions “not embarrassing.”

class Slice {
  static of(object, from = 0, to = Infinity) {
    if (object instanceof this) {
      from = normalizedFrom(object, from, length);
      to = normalizedTo(object, from, to);

      return new Slice(object.array, object.from + from, to - from);
    }
    if (object instanceof Array) {
      from = normalizedFrom(object, from, length);
      to = normalizedTo(object, from, to);

      return new this(object, from, to - from);
    }
    if (typeof object[Symbol.iterator] === 'function') {
      return this.of([...object], from, to);
    }
  }

  // ...
}

function sum (array) {
  return sumOfSlice(Slice.of(array), 0);

  function sumOfSlice (remaining, runningTotal) {
    if (remaining.length === 0) {
      return runningTotal;
    } else {
      const first = remaining[0];
      const rest = remaining.slice(1);

      return sumOfSlice(rest, runningTotal + first);
    }
  }
}

const oneToSix = [1, 2, 3, 4, 5, 6];

sum(oneToSix)
  //=> 21

No more copying entire arrays! And because our .of static method allows us to create a new slice of something and specify the range being sliced, we can also write our function like this:6

function sum (array) {
  return sumOfSlice(Slice.of(array), 0);

  function sumOfSlice (remaining, runningTotal) {
    if (remaining.length === 0) {
      return runningTotal;
    } else {
      const first = remaining[0];
      const rest = Slice.of(remaining, 1);

      return sumOfSlice(rest, runningTotal + first);
    }
  }
}

const oneToSeven = [1, 2, 3, 4, 5, 6, 7];

sum(oneToSeven)
  //=> 28

Naturally, it’s called of, because we use it to take a slice of some list-like object.


List (2007)


more array-ish behaviour

We didn’t need to implement an iterator, but it should be noted that since it has an iterator, we get a lot of JavaScript array-ish behaviour. For example, in strict mode, the iterator is used when destructuring. So if we want to, we can write:

const a1to5 = [1, 2, 3, 4, 5];
const oneToFive = Slice.of(a1to5);
const [first, ...rest] = oneToFive;

first
  //=> 1
rest
  //=> [2, 3, 4, 5]

Unfortunately, destructuring an iterable with the spread operator always creates a new array in JavaScript, so our Slice class can’t help us make const [first, ...rest] = someSlice; not embarrassing. Iterators work with the spread operator in expressions as well:

const abc = ['a', 'b', 'c'];
const oneTwoThree = Slice.of([1, 2, 3]);

[...abc, ...oneTwoThree]
  //=> ["a", "b", "c", 1, 2, 3]

And they get us for... of loops:

const abc = ['a', 'b', 'c'];

const alphabet = {};

for (const letter of Slice.of(abc)) {
  alphabet[letter] = letter;
}

alphabet
  /=> {a: "a", b: "b", c: "c"}

When we dive deeply into the spec, we uncover Symbol.isConcatSpreadable. Forcing it to be true gets us array spread concatenation behaviour. While we’re at it, we can implement .concat:

class Slice {

  // ...

  concat(...args) {
    const { array, from, length } = this;

    return Slice.of(array.slice(from, length).concat(...args));
  }

  get [Symbol.isConcatSpreadable]() {
    return true;
  }
}

const abc = ['a', 'b', 'c'];
const oneTwoThree = Slice.of([1, 2, 3]);

abc.concat(oneTwoThree)
  //=> ["a", "b", "c", 1, 2, 3]
oneTwoThree.concat(abc)
  //=> [ 1, 2, 3, "a", "b", "c"]

Of course, the biggest array-like behaviour our slices are missing is that we haven’t implemented any of the methods for modifying our slices. We’ll do that in Part II. But before moving on, let’s summarize what we’ve done so far.


ideas


wrapping up

We set out with the purpose of writing some code that would allow us to use JavaScript arrays in a Lisp-like style, without the heavy penalty of making lots and lots of copies. To do that, we implemented structural sharing. We added a Proxy to give our new class indexed access to the elements of our Slice class.

While these techniques are far too heavyweight for a simple task like writing a sum function in the style favoured by Lisp programmers of the 1960s and 1970s, that task was small enough and simple enough to allow us to focus on the implementation of these techniques, rather than on the problem of the domain.

These techniques may seem exotic at first, but they form the basis for high-performance implementation of large data structures. And many other languages, such as Clojure, bake these semantics right in. If JavaScript worked like Clojure, there would be no need to implement a Slice class, because arrays would already implement structural sharing. Calling .slice would be inexpensive, right out of the box.

Until the day that JavaScript gets such data structures in its standard library, we’ll have to Greenspun the functionality ourselves, or use a library such as David Nolen’s Mori.

Next: Structural Sharing and Copy-on-Write Semantics, Part II: Reduce-Reuse-Recycle

(discuss on hacker news and reddit; portions of this essay have previously appeared in the book JavaScript Allongé)


Bonus Hack!

Lisp programmers used car and cdr in intricate ways. Although we’ve only looked at simple lists, cons cells could be used to make trees of arbitrary complexity, and the right sequence of car and cdr invocations could navigate a path to any element or sub-tree.

To facilitate this, Lisp had a system where any function name that started with c, ended with r, and had one or more a or d characters in between was automatically also a function, and it was implemented as if the functions car and cdr were composed in order.

For example, (cadr list) was equivalent to (car (cdr list)), which is the second element. If we wanted to really get Lisp-y, we would implement the same scheme…

This being JavaScript, we’ll hack this idea with a proxy and synthetic properties. That way, we can destructure slices, like this:

const SliceHandler = {
  has (slice, property) {
    if (property in slice) {
      return true;
    }

    if (typeof property === 'symbol') {
      return false;
    }

    const matchInt = property.match(/^\d+$/);
    if (matchInt != null) {
      const i = parseInt(property);

      return slice.has(i);
    }

    const matchCarCdr = property.match(/^c([ad]+)r$/);
    if (matchCarCdr != null) {
      return true;
    }
  },

  get (slice, property) {
    if (property in slice) {
      return slice[property];
    }

    if (typeof property === 'symbol') {
      return;
    }

    const matchInt = property.match(/^\d+$/);
    if (matchInt != null) {
      const i = parseInt(property);
      return slice.at(i);
    }

    const matchCarCdr = property.match(/^c([ad]+)r$/);
    if (matchCarCdr != null) {
      const [, accessorString] = matchCarCdr;
      const accessors = accessorString.split('').map(ad => `c${ad}r`);
      return accessors.reduceRight(
        (value, accessor) => Slice.of(value)[accessor],
        slice);
    }
  }
};

class Slice {

  // ...

  get car() {
    return this.at(0);
  }

  get cdr() {
    return this.slice(1);
  }

}

const oneToFive = Slice.of([1, 2, 3, 4, 5]);

const { car: first, cadr: second, cddr: rest } = oneToFive;

first
  //=> 1
second
  //=> 2
[...rest]
  //=> [3, 4, 5]

The Complete Code


End Notes

  1. Lisp is still very much alive, and one of the most interesting and exciting programming languages in use today is Clojure, a Lisp dialect that runs on the JVM, along with its sibling ClojureScript, Clojure that transpiles to JavaScript. Clojure and ClojureScript both make extensive use of structural sharing and copy-on-write semantics to achieve high performance. By default. 

  2. Fun fact: The very first FORTRAN implementation was also written for the IBM 704

  3. When we say, Nobody wants to read code that looks like cons(1, cons(2, cons(3, cons(4, cons(5, null))))), we mean it. Even the Lisp gurus of old didn’t want to deal with that, so in Lisp when you write '(1 2 3 4 5), it is translated directly into a linked list of cons cells. 

  4. All of this code requires the engine to implement strict JavaScript semantics. Some engines can be configured in “loose” mode, where their implementation of things like destructuring may vary from the standard. 

  5. David Wheeler is credited with what is often called The Fundamental Theorem of Software Engineering, “We can solve any problem by introducing an extra level of indirection.” The wording has evolved over time, and the corollary “…except for the problem of too many layers of indirection” is almost always quoted at the same time. 

  6. The version using remaining.slice(1) is going to be more familiar to other programmers, but we will see in Part II how Slice.of(remaining, 1) leads us towards a better understanding of resource ownership.