Quantcast
Channel: Picanteverde
Viewing all articles
Browse latest Browse all 10

JavaScript inheritance and prototype chain search (the end of the inheritance discussion)

$
0
0

This post aims to be the end of the discussion in which JavaScript inheritance method is faster between parasitic and pseudoclassical models.
First of all let present the contenders

PseudoClassical inheritance

The pseudo classical inheritance is the basic method in JavaScript for creating new types, the syntax is quite complex and confusing for example let create a type named “PseudoClass” with just one method named “action1″:

var PseudoClass = function(){
};

PseudoClass.prototype.action1 = function(){
var i;
for(i = 0; i < 10; i += 1){
}
};

now if we want to create a type that inherit it behavior from the previous type and adds ne methods, then we need to do the following

var PseudoSubClass = function(){
};

PseudoSubClass.prototype = new PseudoClass();
PseudoSubClass.prototype.action2 = function(){
var i;
for(i = 0; i < 10; i += 1){
}
};

Now here there is a lot going on, we define the new PseudoSubClass constructor, and the we instantiate the inherited type in the prototype. why? well this is done to connect the prototype chain. When you call a method over an object:

object1.method1();

The JavaScript VM searchs for a property named “method1″ in the “object1″ object, if is not found then it searchs for a property named “method1″ in the prototype of the object “object1″, if is not found then it searchs the property in the prototype of the prototype and so on.

so by using the new operator we connect the prototype of the PseudoClass type to the prototype of the prototype of the PseudoSubClass type.

kind of hard ah?

 

Parasitic Inheritance

This is a much more comfortable model to create new types in JavaScript, is based on the two things JavaScript handles the better, functions and objects. Let create a new type named “parasitic” with a method named “action1″ just reflecting the same behavior as in the previous example

var makeParasitic = function (){
return {
action1:  function(){
var i;
for(i = 0; i < 10; i += 1){
}
}
};
};

The simplicity is overwhelming. we just create a function that creates a new object and then it adds the method to it. Done!.

Now we’ll create a new type inheriting from the previous one

var makeSubParasitic = function(){
var o = makeParasitic();
o.action2 = function(){
var i;
for(i = 0; i < 10; i += 1){
}
};
return o;
};

Just equally simple, we just create a factory function that inside creates a new Parasitic object and adds new behavior to it, avoiding the prototype chain search. This is very important because in this case if you do

object1.method1();

“method1″ will be a property in the “object1″ object and then the function will be executed avoiding the need for a prototypal chain search.

The Tests

There is a maximum that dictates “if you can’t measure it…. you can’t improve it”. Then I created a few jsPerf Test Cases to probe my point. And here is the data collected

Test 1

We’ll compare how much time takes to the VM to search for a method in the prototype chain agains finding the method right in the object.

http://jsperf.com/prototypalchainsearch

The results

test1

as you can see is much more faster to call methods on a parasitic type, than on a pseudoclassical type, this is just because of the prototype chain search of the method

test 1 graph

Test 2

Now let’s compare how about just instantiating objects?

http://jsperf.com/new-operator-instantiation-delay

The results

test 2

It clearly show optimizations an tweaks made on the VM vendor side in favor of the PseudoClassical model, I thinks this is mainly for two reasons. First JavaScript was conceived with this inheritance method in mind so most of the projects out there are using it, and Second it is more simple to optimize because it is just used for type creation, instead functions and objects (parasitic) are much more dynamics and can be used to anything else.

test2 graph

Test 3

Now, let’s compare what happens if we instantiate the object, and the stress the prototype chain search

http://jsperf.com/instantiation-and-method-calling

The results

test3

Here is a much more complex scenario, we are testing both sides, instantiation and method search, take a look how the 78% of difference between PseudoClassical and Parasitical types are drastically reduced to a 48%, and that is calling methods just once, if we stress the method calling for one object we found that the difference is reduce and at some point (500 method search) it is faster to instantiate with parasitic and call methods on that object.

Final Conclusion

It is clear that both methods have their own pros & cons, but if we think just about performance then we clearly see that:

PseudoClassical is a winner for applications that

  • instantiate a lot of objects
  • more object instantiation than methods calling
  • doesn’t call repeatedly methods on them.
  • flat hierarchy of types

Parasitical is a winner for applications with

  • few object instances
  • a lot of method calling
  • repeatedly calling methods
  • deep hierarchy of types

but that is just thinking about performance. If you think about design and elegance of code then the parasitic model is clearly a winner.



Viewing all articles
Browse latest Browse all 10

Trending Articles