How can I extend properly a JS object?
Let's say I have something like this in a file named main.js
:
function obj_name() {}
obj_name.prototype = {
foo : function() { alert('hi!'); },
foo2 : function() { alert('hi again!'); }
}
Now I am trying this way to expand the object in another file extend.js
:
o开发者_JAVA百科bj_name.prototype = {
newfoo : function() { alert('hi #3'); }
}
... but the problem is that it will just work if I code it this way:
obj_name.prototype.newfoo = function() { alert('hi #3'); }
I guess this may be a noob question. I don't even know if this is the proper way to extend an object, but I am freaking out here wondering why does this happen.
Thank you guys in advance.
Another option without jQuery:
var extend = function(destination, source)
{
for (var property in source)
{
if (destination[property] && (typeof(destination[property]) == 'object')
&& (destination[property].toString() == '[object Object]') && source[property])
extend(destination[property], source[property]);
else
destination[property] = source[property];
}
return destination;
}
var a = {a: 'test'}; // original
var b = {newFoo: function() { alert('hi #3'); }}; // the addition
extend(a, b); // extend it
a.newFoo(); // call the added property
This is because on the row
obj_name.prototype = {
newfoo : function() { alert('hi #3'); }
}
you create a new prototype object, deleting privous contents. It is as if you've said
var a = {};
when you extend object like
obj_name.prototype.newfoo = function() { alert('hi #3'); }
it just add a new property (newfoo ) to the object tree keeping existing contents untouched. This is why it works
HTH
Ivo Stoykov
In the first way, you are replacing the prototype with a new one (overwriting what was there before). In the second way, you are adding a new member to the prototype (thus expanding it).
There is another method: a library that has an extend
method or similar (which basically wraps what you're doing in your second form in a nice wrapper). For example, in jQuery:
$.extend(obj_name.prototype, {
newfoo : function() { alert('hi #3'); }
}
If you're looking for a simple lightweight library that gives you exactly this: OOP "done right" in javascript, have a look at this: https://github.com/haroldiedema/joii
Source code examples provided in the readme on the github page, as well as these links:
- http://harold.info/projects/joii
- http://haroldiedema.github.io/joii
This library basically allows you to define "classes" as such:
var Person = Class(function() {
this.firstname = "John"
this.surname = "Smith"
this.role= "Developer"
this.getInfo = function() {
return this.firstname + ' ' + this.surname + ' is ' + this.role;
};
});
var AnotherPerson = Class({ extends: Person }, function() {
this.firstname = "Bob";
});
var p = new AnotherPerson();
console.log(p.getInfo());
// Bob Smith is Developer
Edit
To take your code as an example but transformed into JOII-compatible code, it would look exactly like this:
var obj_name = Class(function() {
this.foo = function() { alert('hi!'); };
this.foo2 = function() { alert('hi again!'); };
};
var obj_name2 = Class({ extends: obj_name }, function() {
this.newfoo = function() { alert('hi #3'); };
});
var o = new obj_name2();
o.foo(); // hi!
o.newfoo(); // hi #3
Or use it as a mix-in:
var o = new obj_name();
o.mixin(obj_name2);
o.newfoo(); // hi #3
Or the other way around, using a "trait".
// the "uses" option basically copies content from the given object to the scope of your "class", solving the horizontal code-reuse problem.
var obj_name = Class({ uses: [obj_name2], function() {
this.foo = function() { alert('hi!'); };
this.foo2 = function() { alert('hi again!'); };
});
var o = new obj_name();
o.newfoo(); // hi #3
精彩评论