This is a purely theoretical question.
I am learing javascript from 'you don't know js', and i was stuck on the implementation of the bind
function in JS. consider the following code:
function foo(something) {
this.a = something;
}
var obj1 = {};
var bar = foo.bind(obj1);
bar(2);
console.log(obj1.a); // 2
var baz = new bar(3);
console.log(obj1.a); // 2
console.log(baz.a); // 3
This is a purely theoretical question.
I am learing javascript from 'you don't know js', and i was stuck on the implementation of the bind
function in JS. consider the following code:
function foo(something) {
this.a = something;
}
var obj1 = {};
var bar = foo.bind(obj1);
bar(2);
console.log(obj1.a); // 2
var baz = new bar(3);
console.log(obj1.a); // 2
console.log(baz.a); // 3
In the above snippet, we bind foo()
to obj1
, so the this
in foo()
belongs to obj1
and that's why obj1.a
bees 2
when we call bar(2)
. but the new
operator is able to take precedence and obj1.a
does not change even when bar(3)
is called with new
.
Below is the polyfill provided by the MDN page for bind(..)
:
if (!Function.prototype.bind) {
Function.prototype.bind = function(oThis) {
if (typeof this !== "function") {
// closest thing possible to the ECMAScript 5
// internal IsCallable function
throw new TypeError( "Function.prototype.bind - what " +
"is trying to be bound is not callable"
);
}
var aArgs = Array.prototype.slice.call( arguments, 1 ),
fToBind = this,
fNOP = function(){},
fBound = function(){
return fToBind.apply(
(
this instanceof fNOP &&
oThis ? this : oThis
),
aArgs.concat( Array.prototype.slice.call( arguments ) )
);
}
;
fNOP.prototype = this.prototype;
fBound.prototype = new fNOP();
return fBound;
};
}
The part that's allowing new overriding according to the book, is:
this instanceof fNOP &&
oThis ? this : oThis
// ... and:
fNOP.prototype = this.prototype;
fBound.prototype = new fNOP();
so , now the main point. according to the book: "We won't actually dive into explaining how this trickery works (it's plicated and beyond our scope here), but essentially the utility determines whether or not the hard-bound function has been called with new (resulting in a newly constructed object being its this), and if so, it uses that newly created this rather than the previously specified hard binding for this."
how is the logic in the bind()
function allows new
operator to override the hard binding?
- You may find this answer useful. – Mitya Commented Oct 2, 2017 at 15:05
-
Languages have pile-time and runtime semantics that are defined and implemented. When a JS program is parsed and piled, calls with the
new
operator are found. This operator is known to be used to call a function with the intent of constructing a new object. As such, the specific algorithm(s) defined by the spec for this circumstance are followed. Part of that, like any call, is settingthis
to the proper value. There's no real magic. The language implementation is just following rules. – llama Commented Oct 2, 2017 at 15:13 - Also, keep in mind that polyfills are "best effort" implementations written purely in JS. They have no access to the language implementation's internals, and so are sometimes unable to perfectly represent the requirements of the spec. – llama Commented Oct 2, 2017 at 15:14
-
Here is a simple example
function Foo(x) { console.log(this instanceof Foo); }
->Foo(4)
= falsenew Foo(4)
= true. – Keith Commented Oct 2, 2017 at 15:14 -
@Utkanos that was really helpful but still didn't explain how is
new
able to take precedence over hard binding – vikrant Commented Oct 2, 2017 at 15:17
3 Answers
Reset to default 8First, it is important to understand the difference between an object's prototype (represented the spec as [[Prototype]]
and accessible via the function Object.getPrototypeOf
or the deprecated __proto__
property) and the property on a function whose name is prototype
. Every function has a property named prototype
which is used when the function is called with new
.
When you call a function with new
, that function is supplied with a this
value set to a newly-constructed object whose prototype (i.e., [[Prototype]]
) is set to the prototype
property of the function being called. That is, when you call new Foo()
, then when the code inside Foo
is run, the this
value will be an object of the form
{ [[Prototype]]: Foo.prototype }
Let's briefly meet the cast of variables:
fToBind
is the function being bound: forfoo.bind(...)
,foo
isfToBind
.fBound
is the bound version offToBind
; it is the returned value of thebind
operation.fBound
acts like a gatekeeper for the originalfToBind
function, and decides whatthis
valuefToBind
gets when it is invoked.oThis
is the first argument supplied tobind
, i.e., the object being bound to the function'sthis
.fNOP
is a function whoseprototype
property is set tofToBind.prototype
fBound.prototype = new fNOP()
causes these to be true:Object.getPrototypeOf(fBound.prototype) === fNOP.prototype Object.getPrototypeOf(fBound.prototype) === fToBind.prototype
When fBound
is called with new
, then the this
that is supplied to fBound
is of the form
{ [[Prototype]]: fBound.prototype }
and fBound.prototype
is an object of the form
{ [[Prototype]]: fNOP.prototype }
making the full form of this
equivalent to
{ [[Prototype]]: { [[Prototype]]: fNOP.prototype } }
So, fNOP.prototype
is in the prototype chain of the newly-created this
object when fBound
is called with new
. That's exactly what the object instanceof constructor
operation tests for:
The
instanceof
operator tests the presence ofconstructor.prototype
inobject
's prototype chain.
The order of operations between &&
and ternary here is:
(this instanceof fNOP && oThis) ? this : oThis
If this
has fNOP.prototype
in its prototype chain and the original bind
call was given a truthy first argument to bind to the function, then use the natrually-created this
supplied to fBound
when it is called with new
and supply that to fToBind
instead of the bound this
.
new
takes precedence over a bound this
value because that's how the language was defined.
First you have a function. Then you bind a this
value, and call it normally. As expected, the value of this
is the bound value.
Then you call that same function with new
, and your value is overridden. Why? Because calls using new
are instructed by the language design, and therefore by the language implementation, to ignore the bound this
value, and replace it with the new object being constructed.
A language implementation is just a program. And like any other program, it follows rules. So the rule in this case is that new
gets to dictate the value of this
irrespective of any bound value.
I guess you're just asking about how they got the polyfill to work.
In that polyfill, fNOP
is a no-op function (does nothing when called) that is used simply to have its .prototype
inserted into the prototype chain of the returned fBound
function. That's done here:
fNOP.prototype = this.prototype;
fBound.prototype = new fNOP();
So then when the fBound
function (the one that gets returned to the caller of .bind()
) is invoked, the instanceof
operator can check the this
value inside the fBound
function to see if that value is an instance of fNOP
. And if it is, it infers that new
was used.
This works (sort of) because instanceof
will start at the object it's given on its left hand side, and search the prototype chain to see if any of them are the same object as the .prototype
object of the function on its right side. So if new
was invoked, the this
value will be a new object that has fNOP.prototype
in its prototype chain because of the setup that was performed as shown above.
However, this isn't a perfect means of testing. For example, the .call()
method could have been used to set the this
value of the call to some other instance of the fBound
function. So it will look like new
was used even when it wasn't, and as such, the bound value for this
will not be used as the this
value of the call to the original function.