Advanced JavaScript: Memoization, Promises, and Event Loop

September 25, 2024 (3mo ago)

Advanced JavaScript: Memoization, Promises, and Event Loop

Understanding JavaScript beyond the basics allows you to write more optimized and efficient code, especially in asynchronous environments. In this article, we'll explore memoization, Promises, and how the event loop works in JavaScript.

Memoization

Memoization is an optimization technique that stores the results of expensive function calls and returns the cached result when the same inputs occur again. It improves performance by avoiding redundant calculations.

function memoize(fn) {
  const cache = {};
  return function(...args) {
    const key = JSON.stringify(args);
    if (cache[key]) {
      return cache[key];
    }
    const result = fn(...args);
    cache[key] = result;
    return result;
  };
}
 
const slowFunction = (num) => {
  console.log("Called with", num);
  return num * 2;
};
 
const memoizedFunction = memoize(slowFunction);
console.log(memoizedFunction(5)); // Called with 5, returns 10
console.log(memoizedFunction(5)); // Returns 10, without calling slowFunction again

Memoization is commonly used in recursive algorithms like the Fibonacci sequence to avoid redundant calculations.

Promises

Promises represent the eventual completion (or failure) of an asynchronous operation and its resulting value. Promises help manage asynchronous code in a cleaner, more readable manner than traditional callbacks.

const fetchData = () => {
  return new Promise((resolve, reject) => {
    setTimeout(() => {
      const success = true;
      if (success) {
        resolve("Data fetched successfully");
      } else {
        reject("Failed to fetch data");
      }
    }, 2000);
  });
};
 
fetchData()
  .then((data) => console.log(data))
  .catch((error) => console.error(error));

Here, the fetchData function returns a promise that resolves after 2 seconds. Promises make it easier to handle asynchronous logic with .then() and .catch() rather than dealing with deeply nested callback functions (callback hell).

The Event Loop

The event loop is what enables JavaScript to be asynchronous. It allows the execution of multiple operations without blocking the main thread, making JavaScript non-blocking.

console.log("Start");
 
setTimeout(() => {
  console.log("Timeout");
}, 0);
 
console.log("End");

In the example above, even though setTimeout has a delay of 0, the message "Timeout" is logged after "End" because of the event loop. The event loop places asynchronous operations like setTimeout in a callback queue, which gets executed after the main execution stack is empty.

Conclusion

Memoization, Promises, and the event loop are critical for writing efficient JavaScript. By using memoization, you can optimize performance; by mastering Promises, you can handle asynchronous tasks more elegantly; and understanding the event loop helps in debugging and writing non-blocking code.