I'm actively learning javascript, and I came across the following statement:
Object.prototype.toString.call([]);
And I don't know what it means or what it does.
I have a vague understanding of .call
, in that it allows you to call a method in the context of a different object (I think), but I am having a hard time understanding what role the .call()
function is playing in the above statement. So I was wondering if anyone could explain what .call()
is doing here?
Thanks!!
The call() allows for a function/method belonging to one object to be assigned and called for a different object.
It's a new feature that introduced in ES6 and is called arrow function. The left part denotes the input of a function and the right part the output of that function.
To call a function inside another function, define the inner function inside the outer function and invoke it. When using the function keyword, the function gets hoisted to the top of the scope and can be called from anywhere inside of the outer function.
The CALL statement transfers control from one object program to another within the run unit. The program containing the CALL statement is the calling program; the program identified in the CALL statement is the called subprogram.
The call
method sets the this
value of the invoked function to the object passed as first argument, in your example, you are executing the Object.prototype.toString
method on an Array object.
Array objects, have their own toString
method (Array.prototype.toString
) that shadows the one from Object.prototype
, if you call [].toString();
the method on the Array.prototype
will be invoked.
For example:
function test() {
alert(this);
}
test.call("Hello"); // alerts "Hello"
Another example:
var alice = {
firstName: 'Alice',
lastName: 'Foo',
getName: function () {
return this.firstName + ' ' + this.lastName;
}
};
var bob = {
firstName: 'Bob',
lastName: 'Bar',
};
alice.getName.call(bob); // "Bob Bar"
In the above example, we use the Alice's getName
method on the Bob's object, the this
value points to bob
, so the method works just as if it were defined on the second object.
Now let's talk about the Object.prototype.toString
method. All native objects in JavaScript contain an internal property called [[Class]]
this property contains a string value that represents the specification defined classification of an object, the possible values for native objects are:
"Object"
"Array"
"Function"
"Date"
"RegExp"
"String"
"Number"
"Boolean"
"Error"
for error objects such as instances of ReferenceError
, TypeError
, SyntaxError
, Error
, etc"Math"
for the global Math
object"JSON"
for the global JSON object defined on the ECMAScript 5th Ed. spec."Arguments"
for the arguments
object (also introduced on the ES5 spec.)"null"
(introduced just a couple of days ago in the ES5 errata)"undefined"
As I've said before that property is internal, there is no way to change it, the specification doesn't provide any operator or built-in function to do it, and the only way you can access its value is through the Object.prototype.toString
method.
This method returns a string formed by:
"[object " + this.[[Class]] + "]"
Only for expository purposes because [[Class]]
cannot be accessed directly.
For example:
Object.prototype.toString.call([]); // "[object Array]"
Object.prototype.toString.call(/foo/); // "[object RegExp]"
Object.prototype.toString.call({}); // "[object Object]"
Object.prototype.toString.call(new Date); // "[object Date]"
// etc...
This is really useful to detect the kind of an object in a safe way, for detecting array objects, it's the most widely used technique:
function isArray(obj) {
return Object.prototype.toString.call(obj) == '[object Array]';
}
It might be tempting to use the instanceof
operator, but that way will lead to problems if you work on cross-frame environments, because an array object created on one frame, will not be instanceof
the Array
constructor of another.
The above method will work without any problems, because the object will contain the value of its [[Class]]
internal property intact.
See also:
instanceof
considered harmful (or how to write a robust isArray
)Object.prototype.toString
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With