HomeAboutBlog

What are closures

A closure is the combination of a function bundled together (enclosed) with references to its surrounding state (the lexical environment). In other words, a closure gives you access to an outer function's [[scope|lang.js.func.scope]] from an inner function. In JavaScript, closures are created every time a function is created, at function creation time.

Lexical Scoping

Consider the following example code:

function init() {
  var name = 'Dan' // name is a local variable created by init
  function displayName() {
    // displayName() is the inner function, a closure
    alert(name) // use variable declared in the parent function
  }
  displayName()
}
init()

init() creates a local variable called name and a function called displayName(). The displayName() function is an inner function that is defined in init() and is available only withing the body of the init() function. Not that the dispalyName() function has no local variables of its own. However, since inner function have access to the variables of outer functions, displayName() can access the variable name declared in the parent function init().

Closure Example

Cosider the following function:

function makeFunc() {
  var name = 'Dan'
  function displayName() {
    alert(name)
  }
  return displayName
}
const myFunc = makeFunc()
myFunc()

Running this code has exactly the same effect as the previous example of the init() function above. What's different (and interesting) is that the displayName() inner function is returned from the outer function before being executed.

At first glance, it might seem unintuitive that this code still works. In some programming languages, the local variables within a function exist for just the duration of that functios execution. Once makeFunc() finishes executing, you might expect that the name variable would no longer be accessible. However, because the code still works as expected, this is obviously not the case in JavaScript.

The reason is that functions in JavaScript form closures. A closure is the combination of the function and the lexical environment withing which that function was declared. This environment consists of any local variables that were in-scope at the time the closure was created. In this case, myFunc is a reference to the instance of the function displayName that is created when makeFunc is run. The instance of displayName maintains a reference to its lexical environment, within which the variable name exists. For this reason, when myFunc is invoked, the variable name remains available for use, and "Dan" gets passed to alert

Here's a slightly more interesting example - a makeAdder function:

function makeAdder(x) {
  return function (y) {
    return x + y
  }
}

const add5 = makeAdder(5)
const add10 = makeAdder(10)

console.log(add5(2)) // 7
console.log(add10(2)) // 12

In this example, we have defined a function makeAdder(x), that takes a single argument x, and returns a new function. The function it returns takes a single argument y, and returns the sum of x and y.

In essence, makeAdder is a fucntion factory. It creates functions that can add a specific value to their argument. In the above example, the function factory creates to new functions - one that adds five to its argument, and one that adds 10.

add5 and add10 are both closures. They share the same function body definition, but store different lexical environments. In add5's lexical environment, x is 5, while in the lexical environment for add10, x is 10.