JavaScript objects (with and without valueOf)
On the JavaScript Telegram channel, they had a "challenge" quiz post that puzzled about 60% of quiz takers (including myself!)
They posted this snippet:
let person = {
name: 'Alice',
age: 30,
valueOf: function() {
return this.age;
}
};
let result = person + 10;
console.log(result);
They asked "What is the output?" with a multiple choice:
[object Object]10
undefined
30
40
All of them seemed like feasible responses, although 30
to me felt the least likely. I guessed the first weird one. As stated earlier, I was incorrect. The correct answer was actually 40
. (Now you might wonder why I even thought it was 30
!)
"What I actually first thought!"
So, it seemed to me that, when we add a number to an object type, JavaScript looks through the object's properties and performs the arithmetic operator on a property that matches the data type. In this case, it was age
, which was 30
. Add 10
to that and we have, of course, a result of 40
.
I actually was quite wrong but continued to experiment to figure it out!
"So what is the person
in the result
?"
So I thought of playing around further and decided to add another property with a number type value:
let person = {
name: 'Alice',
age: 30,
salary: 50000,
valueOf: function() {
return this.age;
}
};
let result = person + 10000;
console.log(result);
In that case, we would actually get 10030
as the result
. The person object seemed to be casted into the number of the person object's first property. It seemed to ignore any other property, even if any of them had number
as a data type.
I looked further to increase my understanding...
"So is person
just the first numeric property?"
So, I tried string concatenation:
let person = {
name: 'Alice',
age: 30,
valueOf: function() {
return this.age;
}
};
let result = person + ' Springs';
console.log(result);
I thought I would get Alice Springs
. I got 30 Springs
but I realized that I didn't even see the person's valueOf
method property!!!
Looking it up, the valueOf
method has this "built-in" characteristic that returns whatever primitive value we tell it to return. (In the case above, that was the age
!)
"It's the valueOf
, stupid!"
So, I tried changing the return
to this.name
:
let person = {
name: 'Alice',
age: 30,
valueOf: function() {
return this.name;
}
};
let result = person + ' Springs';
console.log(result);
...to yield a result
of Alice Springs
!
So, this means that the valueOf
property is a method (i.e. a function
within an object). It has a special property name that will cast its parent object's variable into any primitive value (string, number, boolean, undefined, null, bigint, symbol), via its return
statement!
Okay, what was that?
In the example above, the valueOf
method basically turns person
into this.name
(or whatever will come after the return
statement!)
Now, I fully understood it. This cements the notion that just simply knowing the answer, does not mean understanding it!
"So, if we don't use valueOf
?"
Now, if we do not use valueOf
, we will get a different result
:
let person = {
name: 'Alice',
age: 30,
};
let result = person + 10;
console.log(result);
The person
variable simply gets stringified into [Object object]
.
So, a simple string concatenation occurs: the result
will yield "[Object object]10
"!
Summary
valueOf
exists as a special property name of an object, e.g.parent
- it allows us to write a method that returns a primitive value (i.e. a string, number, boolean, etc.)
- every time we reference
parent
, we will get the value invalueOf
's return statement
- without
valueOf
we would simply get "[Object object]
" every time weconsole.log
an object variable
Key takeaway
Knowing just the answer does not mean understanding it!