Skip to content

Commit 507478e

Browse files
committed
stream: pre-allocate _events
1 parent 4ddb263 commit 507478e

File tree

6 files changed

+55
-4
lines changed

6 files changed

+55
-4
lines changed

lib/events.js

+12-2
Original file line numberDiff line numberDiff line change
@@ -87,6 +87,7 @@ const {
8787

8888
const kCapture = Symbol('kCapture');
8989
const kErrorMonitor = Symbol('events.errorMonitor');
90+
const kShapeMode = Symbol('shapeMode');
9091
const kMaxEventTargetListeners = Symbol('events.maxEventTargetListeners');
9192
const kMaxEventTargetListenersWarned =
9293
Symbol('events.maxEventTargetListenersWarned');
@@ -344,6 +345,9 @@ EventEmitter.init = function(opts) {
344345
this._events === ObjectGetPrototypeOf(this)._events) {
345346
this._events = { __proto__: null };
346347
this._eventsCount = 0;
348+
this[kShapeMode] = false;
349+
} else {
350+
this[kShapeMode] = true;
347351
}
348352

349353
this._maxListeners = this._maxListeners || undefined;
@@ -686,9 +690,13 @@ EventEmitter.prototype.removeListener =
686690
return this;
687691

688692
if (list === listener || list.listener === listener) {
689-
if (--this._eventsCount === 0)
693+
this._eventsCount -= 1;
694+
695+
if (this[kShapeMode]) {
696+
events[type] = undefined;
697+
} else if (this._eventsCount === 0) {
690698
this._events = { __proto__: null };
691-
else {
699+
} else {
692700
delete events[type];
693701
if (events.removeListener)
694702
this.emit('removeListener', type, list.listener || listener);
@@ -750,6 +758,7 @@ EventEmitter.prototype.removeAllListeners =
750758
else
751759
delete events[type];
752760
}
761+
this[kShapeMode] = false;
753762
return this;
754763
}
755764

@@ -762,6 +771,7 @@ EventEmitter.prototype.removeAllListeners =
762771
this.removeAllListeners('removeListener');
763772
this._events = { __proto__: null };
764773
this._eventsCount = 0;
774+
this[kShapeMode] = false;
765775
return this;
766776
}
767777

lib/internal/streams/duplex.js

+17
Original file line numberDiff line numberDiff line change
@@ -63,6 +63,23 @@ function Duplex(options) {
6363
if (!(this instanceof Duplex))
6464
return new Duplex(options);
6565

66+
this._events ??= {
67+
close: undefined,
68+
error: undefined,
69+
prefinish: undefined,
70+
finish: undefined,
71+
drain: undefined,
72+
data: undefined,
73+
end: undefined,
74+
pause: undefined,
75+
resume: undefined,
76+
readable: undefined,
77+
pipe: undefined,
78+
unpipe: undefined,
79+
[destroyImpl.kConstruct]: undefined,
80+
[destroyImpl.kDestroy]: undefined,
81+
};
82+
6683
this._readableState = new Readable.ReadableState(options, this, true);
6784
this._writableState = new Writable.WritableState(options, this, true);
6885

lib/internal/streams/readable.js

+14
Original file line numberDiff line numberDiff line change
@@ -316,6 +316,20 @@ function Readable(options) {
316316
if (!(this instanceof Readable))
317317
return new Readable(options);
318318

319+
this._events ??= {
320+
close: undefined,
321+
error: undefined,
322+
data: undefined,
323+
end: undefined,
324+
pause: undefined,
325+
resume: undefined,
326+
readable: undefined,
327+
pipe: undefined,
328+
unpipe: undefined,
329+
[destroyImpl.kConstruct]: undefined,
330+
[destroyImpl.kDestroy]: undefined,
331+
};
332+
319333
this._readableState = new ReadableState(options, this, false);
320334

321335
if (options) {

lib/internal/streams/writable.js

+10
Original file line numberDiff line numberDiff line change
@@ -382,6 +382,16 @@ function Writable(options) {
382382
if (!(this instanceof Writable))
383383
return new Writable(options);
384384

385+
this._events ??= {
386+
close: undefined,
387+
error: undefined,
388+
prefinish: undefined,
389+
finish: undefined,
390+
drain: undefined,
391+
[destroyImpl.kConstruct]: undefined,
392+
[destroyImpl.kDestroy]: undefined,
393+
};
394+
385395
this._writableState = new WritableState(options, this, false);
386396

387397
if (options) {

test/parallel/test-readline-interface.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,7 @@ class FakeInput extends EventEmitter {
4444
function isWarned(emitter) {
4545
for (const name in emitter) {
4646
const listeners = emitter[name];
47-
if (listeners.warned) return true;
47+
if (listeners && listeners.warned) return true;
4848
}
4949
return false;
5050
}

test/parallel/test-readline-promises-interface.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ class FakeInput extends EventEmitter {
2222
function isWarned(emitter) {
2323
for (const name in emitter) {
2424
const listeners = emitter[name];
25-
if (listeners.warned) return true;
25+
if (listeners && listeners.warned) return true;
2626
}
2727
return false;
2828
}

0 commit comments

Comments
 (0)