Closures Explained: How Functions Remember Their Scope

Closures Explained: How Functions Remember Their Scope

A function in JavaScript remembers the scope it was created in, even after that scope has finished executing. Learn what closures are, why the loop bug happens, and how to use them in practice.

Martin Ferret

Martin Ferret

May 26, 2026

A closure is a function that has access to the variables of its outer scope, even after that outer function has returned. This is not a trick or a special syntax. It is how JavaScript scope works by design.

A First Example

Consider a function that returns another function. The inner function references a variable from the outer one:

      function makeGreeter(name) {
  return function() {
    console.log('Hello, ' + name);
  };
}

const greetJohn = makeGreeter('John');
greetJohn(); // → Hello, John

    

When makeGreeter is called, it returns a function and its execution ends. Yet greetJohn still has access to name. That is a closure: the returned function carries its surrounding scope with it.

Why It Works: Lexical Scope

JavaScript uses lexical scoping: a function's scope is determined by where it is written in the source code, not where it is called from. When a function is defined inside another function, it always has access to the outer function's variables, regardless of when or where it is eventually called.

      function outer() {
  const message = 'I am from outer';

  function inner() {
    console.log(message); // accesses outer's variable
  }

  return inner;
}

const fn = outer();
fn(); // → I am from outer

    

The variable message is not copied into inner. The inner function holds a reference to the outer scope. As long as inner exists, that scope stays alive.

Closures Capture References, Not Values

This is where closures often surprise developers. The closure does not take a snapshot of the variable at the time the function is created. It retains a live reference to it. If the variable changes, the closure sees the new value:

      function makeCounter() {
  let count = 0;

  return function() {
    count++;
    return count;
  };
}

const counter = makeCounter();
console.log(counter()); // → 1
console.log(counter()); // → 2
console.log(counter()); // → 3

    

Each call to counter() increments the same count variable. The closure keeps it alive and private. Nothing outside makeCounter can read or modify count directly.

The Classic Loop Bug

The most common closure pitfall involves loops and var. Because var is function-scoped, all iterations of the loop share the same variable:

      for (var i = 0; i < 3; i++) {
  setTimeout(function() {
    console.log(i);
  }, 1000);
}
// → 3, 3, 3

    

By the time the callbacks fire, the loop has finished and i is 3. All three closures reference the same i.

The fix is to use let, which creates a new binding for each iteration:

      for (let i = 0; i < 3; i++) {
  setTimeout(function() {
    console.log(i);
  }, 1000);
}
// → 0, 1, 2

    

Practical Uses of Closures

Closures are not just a language curiosity. They are the foundation of several everyday patterns in JavaScript.

Private state. A closure can hold data that is inaccessible from the outside, simulating private variables:

      function createWallet(initialBalance) {
  let balance = initialBalance;

  return {
    deposit(amount) { balance += amount; },
    withdraw(amount) { balance -= amount; },
    getBalance() { return balance; }
  };
}

const wallet = createWallet(100);
wallet.deposit(50);
console.log(wallet.getBalance()); // → 150
// balance is not accessible from outside

    

Function factories. Closures let you generate specialized functions from a generic one:

      function multiplier(factor) {
  return (n) => n * factor;
}

const double = multiplier(2);
const triple = multiplier(3);

console.log(double(5)); // → 10
console.log(triple(5)); // → 15

    

A closure is not a design pattern or an advanced trick. It is a direct consequence of how JavaScript handles scope: functions carry their environment with them.

Understanding closures means understanding why counters hold state, why loop callbacks behave unexpectedly with var, and why function factories work at all. Once the model is clear, a large part of JavaScript starts to make sense.

More certificates.dev articles

Get the latest news and updates on developer certifications. Content is updated regularly, so please make sure to bookmark this page or sign up to get the latest content directly in your inbox.

Looking for Certified Developers?

We can help you recruit Certified Developers for your organization or project. The team has helped many customers employ suitable resources from a pool of 100s of qualified Developers.

Let us help you get the resources you need.

Contact Us
Customer Testimonial for Hiring
like a breath of fresh air
Everett Owyoung
Everett Owyoung
Head of Talent for ThousandEyes
(a Cisco company)