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
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 2
Now let’s compare how about just instantiating objects?
http://jsperf.com/new-operator-instantiation-delay
The results
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.
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
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.
