perf(Change Detection): remove the usage of getters/setters

- Firefox is 2.4x faster (90 vs 221ms)
- Chrome is 24% slower (15.5 vs 12.5ms)

Chrome is still 5.8x faster than Firefox
This commit is contained in:
Victor Berchet 2014-12-04 16:36:49 +01:00
parent 2c4a2f5158
commit 68da0012cc
3 changed files with 35 additions and 35 deletions

View File

@ -130,11 +130,11 @@ export class Record {
this._mode = protoRecord._mode; this._mode = protoRecord._mode;
// Return early for collections, further init delayed until updateContext() // Return early for collections, further init delayed until updateContext()
if (this.isCollection) return; if (this.isCollection()) return;
this.currentValue = _fresh; this.currentValue = _fresh;
var type = this.type; var type = this.getType();
if (type === RECORD_TYPE_CONST) { if (type === RECORD_TYPE_CONST) {
this.funcOrValue = protoRecord.funcOrValue; this.funcOrValue = protoRecord.funcOrValue;
@ -159,22 +159,22 @@ export class Record {
} }
} }
// todo(vicb): getter / setters are much slower than regular methods // getters & setters perform much worse on some browsers
// todo(vicb): update the whole code base // see http://jsperf.com/vicb-getter-vs-function
get type():int { getType():int {
return this._mode & RECORD_TYPE_MASK; return this._mode & RECORD_TYPE_MASK;
} }
set type(value:int) { setType(value:int) {
this._mode = (this._mode & ~RECORD_TYPE_MASK) | value; this._mode = (this._mode & ~RECORD_TYPE_MASK) | value;
} }
get disabled():boolean { isDisabled():boolean {
return (this._mode & RECORD_FLAG_DISABLED) === RECORD_FLAG_DISABLED; return (this._mode & RECORD_FLAG_DISABLED) === RECORD_FLAG_DISABLED;
} }
isEnabled():boolean { isEnabled():boolean {
return ! this.disabled; return !this.isDisabled();
} }
_setDisabled(value:boolean) { _setDisabled(value:boolean) {
@ -210,11 +210,11 @@ export class Record {
this._setDisabled(true); this._setDisabled(true);
} }
get isImplicitReceiver():boolean { isImplicitReceiver():boolean {
return (this._mode & RECORD_FLAG_IMPLICIT_RECEIVER) === RECORD_FLAG_IMPLICIT_RECEIVER; return (this._mode & RECORD_FLAG_IMPLICIT_RECEIVER) === RECORD_FLAG_IMPLICIT_RECEIVER;
} }
get isCollection():boolean { isCollection():boolean {
return (this._mode & RECORD_FLAG_COLLECTION) === RECORD_FLAG_COLLECTION; return (this._mode & RECORD_FLAG_COLLECTION) === RECORD_FLAG_COLLECTION;
} }
@ -223,7 +223,7 @@ export class Record {
} }
check():boolean { check():boolean {
if (this.isCollection) { if (this.isCollection()) {
return this._checkCollection(); return this._checkCollection();
} else { } else {
return this._checkSingleRecord(); return this._checkSingleRecord();
@ -250,7 +250,7 @@ export class Record {
// return whether the content has changed // return whether the content has changed
_checkCollection():boolean { _checkCollection():boolean {
switch(this.type) { switch(this.getType()) {
case RECORD_TYPE_KEY_VALUE: case RECORD_TYPE_KEY_VALUE:
var kvChangeDetector:KeyValueChanges = this.currentValue; var kvChangeDetector:KeyValueChanges = this.currentValue;
return kvChangeDetector.check(this.context); return kvChangeDetector.check(this.context);
@ -266,12 +266,12 @@ export class Record {
return true; return true;
default: default:
throw new BaseException(`Unsupported record type (${this.type})`); throw new BaseException(`Unsupported record type (${this.getType()})`);
} }
} }
_calculateNewValue() { _calculateNewValue() {
switch (this.type) { switch (this.getType()) {
case RECORD_TYPE_PROPERTY: case RECORD_TYPE_PROPERTY:
return this.funcOrValue(this.context); return this.funcOrValue(this.context);
@ -291,7 +291,7 @@ export class Record {
return this.funcOrValue; return this.funcOrValue;
default: default:
throw new BaseException(`Unsupported record type (${this.type})`); throw new BaseException(`Unsupported record type (${this.getType()})`);
} }
} }
@ -304,25 +304,25 @@ export class Record {
this.context = value; this.context = value;
this.enable(); this.enable();
if (this.isCollection) { if (this.isCollection()) {
if (ArrayChanges.supports(value)) { if (ArrayChanges.supports(value)) {
if (this.type != RECORD_TYPE_ARRAY) { if (this.getType() != RECORD_TYPE_ARRAY) {
this.type = RECORD_TYPE_ARRAY; this.setType(RECORD_TYPE_ARRAY);
this.currentValue = new ArrayChanges(); this.currentValue = new ArrayChanges();
} }
return; return;
} }
if (KeyValueChanges.supports(value)) { if (KeyValueChanges.supports(value)) {
if (this.type != RECORD_TYPE_KEY_VALUE) { if (this.getType() != RECORD_TYPE_KEY_VALUE) {
this.type = RECORD_TYPE_KEY_VALUE; this.setType(RECORD_TYPE_KEY_VALUE);
this.currentValue = new KeyValueChanges(); this.currentValue = new KeyValueChanges();
} }
return; return;
} }
if (isBlank(value)) { if (isBlank(value)) {
this.type = RECORD_TYPE_NULL; this.setType(RECORD_TYPE_NULL);
} else { } else {
throw new BaseException("Collection records must be array like, map like or null"); throw new BaseException("Collection records must be array like, map like or null");
} }
@ -333,8 +333,8 @@ export class Record {
return !(this.dest instanceof Record); return !(this.dest instanceof Record);
} }
get isMarkerRecord() { isMarkerRecord():boolean {
return this.type == RECORD_TYPE_MARKER; return this.getType() == RECORD_TYPE_MARKER;
} }
expressionMemento() { expressionMemento() {
@ -357,8 +357,8 @@ export class Record {
if (this.isEnabled()) return this.nextEnabled; if (this.isEnabled()) return this.nextEnabled;
var record = this.next; var record = this.next;
while (isPresent(record) && record.disabled) { while (isPresent(record) && record.isDisabled()) {
if (record.isMarkerRecord && record.recordRange.disabled) { if (record.isMarkerRecord() && record.recordRange.disabled) {
record = record.recordRange.tailRecord.next; record = record.recordRange.tailRecord.next;
} else { } else {
record = record.next; record = record.next;
@ -378,8 +378,8 @@ export class Record {
if (this.isEnabled()) return this.prevEnabled; if (this.isEnabled()) return this.prevEnabled;
var record = this.prev; var record = this.prev;
while (isPresent(record) && record.disabled) { while (isPresent(record) && record.isDisabled()) {
if (record.isMarkerRecord && record.recordRange.disabled) { if (record.isMarkerRecord() && record.recordRange.disabled) {
record = record.recordRange.headRecord.prev; record = record.recordRange.headRecord.prev;
} else { } else {
record = record.prev; record = record.prev;

View File

@ -122,7 +122,7 @@ export class RecordRange {
var lastRecord = this.tailRecord.prev; var lastRecord = this.tailRecord.prev;
_link(lastRecord, record); _link(lastRecord, record);
if (!lastRecord.disabled) { if (!lastRecord.isDisabled()) {
_linkEnabled(lastRecord, record); _linkEnabled(lastRecord, record);
} }
_link(record, this.tailRecord); _link(record, this.tailRecord);
@ -210,8 +210,8 @@ export class RecordRange {
*/ */
findFirstEnabledRecord() { findFirstEnabledRecord() {
var record = this.headRecord.next; var record = this.headRecord.next;
while (record !== this.tailRecord && record.disabled) { while (record !== this.tailRecord && record.isDisabled()) {
if (record.isMarkerRecord && record.recordRange.disabled) { if (record.isMarkerRecord() && record.recordRange.disabled) {
record = record.recordRange.tailRecord.next; record = record.recordRange.tailRecord.next;
} else { } else {
record = record.next; record = record.next;
@ -234,8 +234,8 @@ export class RecordRange {
*/ */
findLastEnabledRecord() { findLastEnabledRecord() {
var record = this.tailRecord.prev; var record = this.tailRecord.prev;
while (record !== this.headRecord && record.disabled) { while (record !== this.headRecord && record.isDisabled()) {
if (record.isMarkerRecord && record.recordRange.disabled) { if (record.isMarkerRecord() && record.recordRange.disabled) {
record = record.recordRange.headRecord.prev; record = record.recordRange.headRecord.prev;
} else { } else {
record = record.prev; record = record.prev;
@ -256,7 +256,7 @@ export class RecordRange {
record != null; record != null;
record = record.next) { record = record.next) {
if (record.isImplicitReceiver) { if (record.isImplicitReceiver()) {
this._setContextForRecord(context, record); this._setContextForRecord(context, record);
} }
} }

View File

@ -212,8 +212,8 @@ export function main() {
record2.disable(); record2.disable();
record3.disable(); record3.disable();
expect(record2.disabled).toBeTruthy(); expect(record2.isDisabled()).toBeTruthy();
expect(record3.disabled).toBeTruthy(); expect(record3.isDisabled()).toBeTruthy();
expect(enabledRecords(rr, recordNames)).toEqual(['record1', 'record4']); expect(enabledRecords(rr, recordNames)).toEqual(['record1', 'record4']);
}); });