I found this problem in a GitHub front-end interview questions collection:
var foo = {n: 1}; var bar = foo; foo.x = foo = {n: 2};
Question: What is the value of foo.x?
The answer is undefined
.
I've done some research and what I understand this problem is (correct me if I'm wrong):
var foo = {n: 1};
declares an object foo
which has property n
equal to 1.var bar = foo;
declares an object bar
which refers to the same object as foo
.foo.x = foo = {n: 2};
which I believe is equal to foo.x = (foo = {n: 2});
foo.x
equals to undefined
. However, the value of bar.x
is the object {n:2}
.If bar
and foo
refer to same object, why did bar.x
get a value while foo.x
is undefined
? What is really happening in foo.x = foo = {n: 2};
?
foo.x = foo = {n: 2};
determines that foo.x
refers to a property x
of the {n: 1}
object, assigns {n: 2}
to foo
, and assigns the new value of foo
– {n: 2}
– to the property x
of the {n: 1}
object.
The important thing is that the foo
that foo.x
refers to is determined before foo
changes.
See section 11.13.1 of the ES5 spec:
Let lref be the result of evaluating LeftHandSideExpression.
Let rref be the result of evaluating AssignmentExpression.
The assignment operator associates right to left, so you get:
foo.x = (foo = {n: 2})
The left hand side is evaluated before the right hand side.