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.
@matthewrobb yeah what i hate about the new ES6 class syntax is that it moves us further from understanding what's actually going on. It makes it seem like
Foo
andBaz
in this example are abstract entities that need to get instantiated. That's the wrong mental model. They are actually real proper objects, linked to each other via their[[Prototype]]
chain, and whenb
is created, it's not instantiating some class entity, but just creating another object and adding to the chain.For years, people didn't understand that
.prototype.blah
type behavior was actually doing "event delegation", and they kept trying to think of it as static inheritance (c++ style), and then being surprised when the "child-effects-the-parent" kind of behavior comes into play, and calling that a design flaw.Then, for a brief couple of years here, we've had a new mental model, that of
Object.create(..)
, which actually makes complete and total semantic matching sense with what's really happening. The only problem is not enough people have embraced it and explained it that way.NOW, we're regressing back to bad semantics, to shut up some of the vocal java-in-javascript crowd, by introducing ES6 classes. It's going to set back the language I think. It we just retrained everyone to think like "3.js", it makes total and complete sense and it actually works.