Skip to content

Instantly share code, notes, and snippets.

@getify
Created November 30, 2012 12:04
Show Gist options
  • Save getify/4175392 to your computer and use it in GitHub Desktop.
Save getify/4175392 to your computer and use it in GitHub Desktop.
rethink how javascript "inheritance" ACTUALLY works...

JavaScript does not have "inheritance" or "prototypal inheritance" or "classes" or any of that jazz... what you've been told, and how you've been taught about it, are a misunderstanding... all this nonsense about constructor functions and new and such... that's all hogwash... well, it's all unnecessary effort, at best.

"Instead... only try to realize the truth... there is no spoon."

What JavaScript does have is "behavior delegation"... and object linking through the "prototype chain"... you merely link two instances of objects together, say Foo and Baz, and say that Baz "delegates" to Foo for any behavior that Baz doesn't own itself but that Foo does own. And thus, any object b (aka, "b is an instance of Baz" in the more confusing terminology) which is linked to Baz, delegates first to Baz, and then to Foo, for behavior.

That's it. Seriously. And function constructors and new and all that stuff, everything you've read before about OO in JS, they're just distractions that lead you to the wrong mental model. And so is the new class cruft (aka "sugar") that's coming in ES6. What's actually useful is the semantics of Object.create() actually exposing how this stuff really works.

See what I mean below, comparing "2.js" (the old school and more confusing way) and "3.js" (the cleaner and "object linking and behavior delegation" way). They both do the same thing. But the latter actually says what it means, while the former masquerades as something it's not.

// "class Foo"
function Foo() {
this.bar = "bar";
}
Foo.prototype.out = function() {
console.log(this.bar);
};
// "class Baz"
function Baz() {
this.baz = "baz";
}
Baz.prototype = new Foo();
Baz.prototype.yes = function() {
this.out();
console.log(this.baz);
};
var b = new Baz();
b.yes(); // "bar" "baz"
// "class Foo"
var Foo = Object.create(null);
Foo.bar = "bar";
Foo.out = function() {
console.log(this.bar);
};
// "class Baz"
var Baz = Object.create(Foo);
Baz.baz = "baz";
Baz.yes = function() {
this.out();
console.log(this.baz);
};
// instead of `var b = new Baz()`
var b = Object.create(Baz);
b.yes(); // "bar" "baz"
@simonzack
Copy link

You can't represent constructors as easily in 3.js. That's what new is for.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment