Bird
Raised Fist0
Node.jsframework~8 mins

Custom event emitter classes in Node.js - Performance & Optimization

Choose your learning style10 modes available

Start learning this pattern below

Jump into concepts and practice - no test required

or
Recommended
Test this pattern10 questions across easy, medium, and hard to know if this pattern is strong
Performance: Custom event emitter classes
MEDIUM IMPACT
This concept affects how efficiently events are handled and propagated in a Node.js application, impacting responsiveness and CPU usage.
Creating a custom event emitter to handle multiple event listeners
Node.js
const { EventEmitter } = require('events');
class GoodEmitter extends EventEmitter {
  // Inherits optimized event handling from Node.js core
}

const emitter = new GoodEmitter();
emitter.on('event', async (data) => {
  // handle event asynchronously
});
Uses Node.js built-in EventEmitter which is optimized for performance, supports async listeners, and handles errors gracefully.
📈 Performance GainNon-blocking event handling; fewer CPU spikes; better error management.
Creating a custom event emitter to handle multiple event listeners
Node.js
class BadEmitter {
  constructor() {
    this.listeners = {};
  }
  on(event, listener) {
    if (!this.listeners[event]) {
      this.listeners[event] = [];
    }
    this.listeners[event].push(listener);
  }
  emit(event, ...args) {
    if (this.listeners[event]) {
      this.listeners[event].forEach(listener => {
        listener(...args);
      });
    }
  }
}
This implementation uses a simple array and calls listeners synchronously without error handling or optimization, causing potential blocking and unhandled exceptions.
📉 Performance CostBlocks event loop during emit; triggers multiple synchronous calls; no batching or async handling.
Performance Comparison
PatternDOM OperationsReflowsPaint CostVerdict
Custom synchronous event emitterN/AN/AN/A[X] Bad
Node.js built-in EventEmitter with async listenersN/AN/AN/A[OK] Good
Rendering Pipeline
Custom event emitters affect the Node.js event loop and callback queue, influencing how quickly events are processed and how responsive the application feels.
Event Loop
Callback Execution
⚠️ BottleneckSynchronous listener execution blocks the event loop, delaying other tasks.
Core Web Vital Affected
INP
This concept affects how efficiently events are handled and propagated in a Node.js application, impacting responsiveness and CPU usage.
Optimization Tips
1Avoid synchronous listener calls that block the event loop.
2Use Node.js built-in EventEmitter for optimized event handling.
3Implement asynchronous listeners to keep the app responsive.
Performance Quiz - 3 Questions
Test your performance knowledge
What is a main performance risk of a custom event emitter that calls listeners synchronously without error handling?
AIt blocks the event loop causing delays in other operations.
BIt reduces memory usage significantly.
CIt improves event handling speed by parallel execution.
DIt automatically batches events to reduce CPU load.
DevTools: Performance
How to check: Run your Node.js app with --inspect flag, open Chrome DevTools, go to Performance panel, record while emitting events, and analyze event loop delays.
What to look for: Look for long tasks blocking the event loop during event emission; shorter tasks indicate better performance.

Practice

(1/5)
1. What is the main purpose of creating a custom event emitter class in Node.js?
easy
A. To replace the need for functions and callbacks
B. To allow different parts of your program to communicate by sending and listening to events
C. To speed up the execution of synchronous code
D. To automatically handle HTTP requests

Solution

  1. Step 1: Understand event emitters

    Event emitters let parts of a program send signals (events) and others listen and react to them.
  2. Step 2: Purpose of custom event emitter classes

    Custom classes extend EventEmitter to organize and manage these events clearly.
  3. Final Answer:

    To allow different parts of your program to communicate by sending and listening to events -> Option B
  4. Quick Check:

    Event communication = C [OK]
Hint: Event emitters enable communication between code parts [OK]
Common Mistakes:
  • Thinking event emitters speed up synchronous code
  • Confusing event emitters with HTTP handling
  • Believing event emitters replace all functions
2. Which of the following is the correct way to listen for an event named data in a custom event emitter instance myEmitter?
easy
A. myEmitter.on('data', callback)
B. myEmitter.emit('data', callback)
C. myEmitter.listen('data', callback)
D. myEmitter.trigger('data', callback)

Solution

  1. Step 1: Identify the method to listen for events

    The on method is used to register a callback for an event.
  2. Step 2: Check other options

    emit triggers events, listen and trigger are not valid EventEmitter methods.
  3. Final Answer:

    myEmitter.on('data', callback) -> Option A
  4. Quick Check:

    Listen with on() = A [OK]
Hint: Use on() to listen, emit() to send events [OK]
Common Mistakes:
  • Using emit() to listen instead of on()
  • Using non-existent methods like listen() or trigger()
  • Confusing event names with method names
3. Consider this code snippet:
const EventEmitter = require('events');
class MyEmitter extends EventEmitter {}
const emitter = new MyEmitter();
emitter.on('greet', name => console.log(`Hello, ${name}!`));
emitter.emit('greet', 'Alice');
What will be printed when this code runs?
medium
A. greet Alice
B. Error: greet event not found
C. Hello, Alice!
D. undefined

Solution

  1. Step 1: Understand event registration

    The on method registers a listener for 'greet' that prints a greeting with the name.
  2. Step 2: Understand event emission

    The emit method triggers 'greet' with argument 'Alice', so the listener runs and prints the message.
  3. Final Answer:

    Hello, Alice! -> Option C
  4. Quick Check:

    Emit triggers listener output = B [OK]
Hint: emit() runs on() listeners with given arguments [OK]
Common Mistakes:
  • Expecting event name or arguments to print directly
  • Confusing emit() with on()
  • Assuming error if no listener exists
4. What is wrong with this custom event emitter code?
const EventEmitter = require('events');
class MyEmitter extends EventEmitter {}
const emitter = new MyEmitter();
emitter.emit('start');
emitter.on('start', () => console.log('Started'));
medium
A. The event name 'start' is invalid
B. emit() cannot be called without arguments
C. You must call super() in the constructor of MyEmitter
D. The event listener is registered after the event is emitted, so it won't run

Solution

  1. Step 1: Check event listener registration timing

    The listener for 'start' is added after the event is emitted, so it misses the event.
  2. Step 2: Validate other options

    emit() can be called without arguments, super() is optional if no constructor, and 'start' is a valid event name.
  3. Final Answer:

    The event listener is registered after the event is emitted, so it won't run -> Option D
  4. Quick Check:

    Listener after emit = no output = D [OK]
Hint: Register listeners before emitting events [OK]
Common Mistakes:
  • Calling emit() before on() listener
  • Thinking emit() needs arguments always
  • Assuming constructor must call super() if none defined
5. You want to create a custom event emitter class that counts how many times an event named ping is emitted. Which code correctly implements this behavior?
hard
A. class PingCounter extends EventEmitter { constructor() { super(); this.count = 0; this.on('ping', () => this.count++); } }
B. class PingCounter extends EventEmitter { constructor() { this.count = 0; this.on('ping', () => this.count++); super(); } }
C. class PingCounter extends EventEmitter { count = 0; on('ping', () => this.count++); }
D. class PingCounter extends EventEmitter { constructor() { super(); this.count = 0; this.emit('ping', () => this.count++); } }

Solution

  1. Step 1: Proper constructor and super() call

    class PingCounter extends EventEmitter { constructor() { super(); this.count = 0; this.on('ping', () => this.count++); } } correctly calls super() first in constructor, required before using this.
  2. Step 2: Correct event listener setup

    class PingCounter extends EventEmitter { constructor() { super(); this.count = 0; this.on('ping', () => this.count++); } } uses this.on('ping', () => this.count++) to increment count on each ping event.
  3. Step 3: Check other options for errors

    class PingCounter extends EventEmitter { constructor() { this.count = 0; this.on('ping', () => this.count++); super(); } } calls this.on before super(), causing error. class PingCounter extends EventEmitter { count = 0; on('ping', () => this.count++); } has invalid syntax outside constructor. class PingCounter extends EventEmitter { constructor() { super(); this.count = 0; this.emit('ping', () => this.count++); } } wrongly uses emit instead of on.
  4. Final Answer:

    class PingCounter extends EventEmitter { constructor() { super(); this.count = 0; this.on('ping', () => this.count++); } } -> Option A
  5. Quick Check:

    super() first, then on() listener = A [OK]
Hint: Always call super() before using this in constructor [OK]
Common Mistakes:
  • Calling this before super() in constructor
  • Using emit() instead of on() to listen
  • Placing on() calls outside constructor or methods