Event Loop and EventEmitter

Event Loop

  • Node.js์—์„  Event๋ฅผ ๋งค์šฐ ๋งŽ์ด ์‚ฌ์šฉํ•˜๊ณ , ์ด ๋•Œ๋ฌธ์— ๋‹ค๋ฅธ ๊ธฐ์ˆ ๋“ค๋ณด๋‹ค ํ›จ์”ฌ ์†๋„๊ฐ€ ๋น ๋ฆ„

  • Node.js ๊ธฐ๋ฐ˜์œผ๋กœ ๋งŒ๋“ค์–ด์ง„ server๊ฐ€ ๊ฐ€๋™๋˜๋ฉด,

    • ๋ณ€์ˆ˜๋“ค์„ initializeํ•˜๊ณ ,

    • ํ•จ์ˆ˜๋ฅผ ์„ ์–ธํ•˜๊ณ ,

    • Event๊ฐ€ ์ผ์–ด๋‚  ๋•Œ ๊นŒ์ง€ ๊ธฐ๋‹ค๋ฆผ

  • Event-Driven application์—์„œ๋Š” event๋ฅผ ๋Œ€๊ธฐํ•˜๋Š” main loop๊ฐ€ ์žˆ์Œ

    • ๊ทธ๋ฆฌ๊ณ  event๊ฐ€ ๊ฐ์ง€๋˜์—ˆ์„ ์‹œ Callback ํ•จ์ˆ˜ ๋ฅผ ํ˜ธ์ถœํ•จ

ff

Event vs Callback Function

  • Callback function์€ ๋น„๋™๊ธฐ์‹ ํ•จ์ˆ˜์—์„œ ๊ฒฐ๊ณผ๋ฅผ ๋ฐ˜ํ™˜ํ•  ๋•Œ ํ˜ธ์ถœ

  • Event Handling์€ observer pattern์— ์˜ํ•ด ์ž‘๋™๋จ

Observer pattern

  • object์˜ ์ƒํƒœ ๋ณ€ํ™”๋ฅผ ๊ด€์ฐฐํ•˜๋Š” ๊ด€์ฐฐ์ž๋“ค, ์ฆ‰ observer๋“ค์˜ ๋ชฉ๋ก์„ object์— ๋“ฑ๋กํ•˜์—ฌ ์ƒํƒœ ๋ณ€ํ™”๊ฐ€ ์žˆ์„ ๋•Œ๋งˆ๋‹ค method ๋“ฑ์„ ํ†ตํ•ด ๊ฐ์ฒด๊ฐ€ ์ง์ ‘ ๋ชฉ๋ก์˜ ๊ฐ observer์—๊ฒŒ ํ†ต์ง€ํ•˜๋„๋ก ํ•˜๋Š” design pattern

  • ์ฃผ๋กœ Distributed event handling system ์„ ๊ตฌํ˜„ํ•˜๋Š” ๋ฐ ์‚ฌ์šฉ๋จ

์ด๋ฒคํŠธ๋ฅผ ๋Œ€๊ธฐํ•˜๋Š” EventListeners ํ•จ์ˆ˜๋“ค์ด observer ์—ญํ• ์„ ํ•จ!

-> Observer๋“ค์ด event๋ฅผ ๊ธฐ๋‹ค๋ฆฌ๋‹ค๊ฐ€, event๊ฐ€ ์‹คํ–‰๋˜๋ฉด event๋ฅผ ์ฒ˜๋ฆฌํ•˜๋Š” ํ•จ์ˆ˜๊ฐ€ ์‹คํ–‰๋จ!

events module & EventEmitter Class

: Node.js์˜ event module๊ณผ EventEmitter Class ๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ event์™€ event handler๋ฅผ bind ํ•  ์ˆ˜ ์žˆ๋‹ค

์‚ฌ์šฉ ๋ฐฉ๋ฒ•

// events module ์‚ฌ์šฉ
var events = require('events')

// EventEmitter object ์ƒ์„ฑ
var eventEmitter = new events.EventEmitter();

// Bind event & EventHandler
eventEmitter.on('eventName', eventHandler);

// Program ์•ˆ์—์„œ event๋ฅผ ๋ฐœ์ƒ์‹œํ‚ค๊ธฐ
eventEmtter.emit('eventName');

ex)

event_handle.js

// events module ์‚ฌ์šฉ
var events = require('events');

// EventEmitter object ์ƒ์„ฑ
var eventEmitter = new events.EventEmitter();

// EventHandler ํ•จ์ˆ˜ ์ƒ์„ฑ
var connectHandler = function connected(){
    console.log("You've connected");

    // data_received event ๋ฐœ์ƒ์‹œํ‚ค๊ธฐ
    eventEmitter.emit('data_received');
}

// connection event์™€ connectHandler ๋ฅผ ์—ฐ๋™
eventEmitter.on('connection', connectHandler);

// data_received event์™€ anonymous function ์—ฐ๋™
// -> ํ•จ์ˆ˜๋ฅผ ๋ณ€์ˆ˜์•ˆ์— ๋‹ด๋Š” ๋Œ€์‹ ์—, .on() method์˜ ์ธ์ž๋กœ ์ง์ ‘ ํ•จ์ˆ˜๋ฅผ ์ „๋‹ฌ
eventEmitter.on('data_received', function() {
    console.log('Data received');
}) 

// connection event ๋ฐœ์ƒ ์‹œํ‚ค๊ธฐ
eventEmitter.emit('connection');

console.log('Program has ended');

Result

$ node event_handle.js 
You've connected
Data received
Program has ended

EventEmitter Class

  • Many objects in a Node emit events

    • Server emits an event each time a peer connects to it

    • An fs.readStream emits an event when the file is opened

  • All objects which emit events are instance of events.EventEmitter

Methods

Sr.No.
Method & Description

1

addListener(event, listener) Adds a listener at the end of the listeners array for the specified event. No checks are made to see if the listener has already been added. Multiple calls passing the same combination of event and listener will result in the listener being added multiple times. Returns emitter, so calls can be chained.

2

on(event, listener) Adds a listener at the end of the listeners array for the specified event. No checks are made to see if the listener has already been added. Multiple calls passing the same combination of event and listener will result in the listener being added multiple times. Returns emitter, so calls can be chained.

3

once(event, listener) Adds a one time listener to the event. This listener is invoked only the next time the event is fired, after which it is removed. Returns emitter, so calls can be chained.

4

removeListener(event, listener) Removes a listener from the listener array for the specified event. Caution โˆ’ It changes the array indices in the listener array behind the listener. removeListener will remove, at most, one instance of a listener from the listener array. If any single listener has been added multiple times to the listener array for the specified event, then removeListener must be called multiple times to remove each instance. Returns emitter, so calls can be chained.

5

removeAllListeners([event]) Removes all listeners, or those of the specified event. It's not a good idea to remove listeners that were added elsewhere in the code, especially when it's on an emitter that you didn't create (e.g. sockets or file streams). Returns emitter, so calls can be chained.

6

setMaxListeners(n) By default, EventEmitters will print a warning if more than 10 listeners are added for a particular event. This is a useful default which helps finding memory leaks. Obviously not all Emitters should be limited to 10. This function allows that to be increased. Set to zero for unlimited.

7

listeners(event) Returns an array of listeners for the specified event.

8

emit(event, [arg1], [arg2], [...]) Execute each of the listeners in order with the supplied arguments. Returns true if the event had listeners, false otherwise.

Class Methods

Sr.No.
Method & Description

1

listenerCount(emitter, event) Returns the number of listeners for a given event.

Events

Sr.No.
Events & Description

1

newListener event โˆ’ String : the event name listener โˆ’ Function : the event handler functionThis event is emitted any time a listener is added. When this event is triggered, the listener may not yet have been added to the array of listeners for the event.

2

removeListener event โˆ’ String The event name listener โˆ’ Function The event handler function This event is emitted any time someone removes a listener. When this event is triggered, the listener may not yet have been removed from the array of listeners for the event.

ex)

EventEmitter.js

var events = require('events');
var eventEmitter = new events.EventEmitter();

// listener #1
var listner1 = function listner1() {
   console.log('listner1 executed.');
}

// listener #2
var listner2 = function listner2() {
   console.log('listner2 executed.');
}

// Bind the connection event with the listner1 function
eventEmitter.addListener('connection', listner1);

// Bind the connection event with the listner2 function
eventEmitter.on('connection', listner2);

var eventListeners = require('events').EventEmitter.listenerCount
   (eventEmitter,'connection');
console.log(eventListeners + " Listner(s) listening to connection event");

// Fire the connection event 
eventEmitter.emit('connection');

// Remove the binding of listner1 function
eventEmitter.removeListener('connection', listner1);
console.log("Listner1 will not listen now.");

// Fire the connection event 
eventEmitter.emit('connection');

eventListeners = require('events').EventEmitter.listenerCount(eventEmitter,'connection');
console.log(eventListeners + " Listner(s) listening to connection event");

console.log("Program Ended.");

Result

$ node EventEmitter.js 
2 Listner(s) listening to connection event
listner1 executed.
listner2 executed.
Listner1 will not listen now.
listner2 executed.
1 Listner(s) listening to connection event
Program Ended.

Last updated

Was this helpful?