Friday, 2 December 2016

JavaScript by reference vs. by value




I'm looking for some good comprehensive reading material on when JavaScript passes something by value and when by reference and when modifying a passed item affects the value outside a function and when not. I'm also interested in when assigning to another variable is by reference vs. by value and whether that follows any different rules than passing as a function parameter.



I've done a lot of searching and find lots of specific examples (many of them here on SO) from which I can start to piece together pieces of the real rules, but I haven't yet found a single, well written document that describes it all.




Also, are there ways in the language to control whether something is passed by reference or by value?



Here are some of the types of questions I want to understand. These are just examples - I'm actually looking to understand the rules the language goes by, not just the answers to specific examples. But, here are some examples:



function f(a,b,c) {
a = 3;
b.push("foo");
c.first = false;
}


var x = 4;
var y = ["eeny", "miny", "mo"];
var z = {first: true};
f(x,y,z);


When are the contents of x, y and z changed outside the scope of f for all the different types?



function f() {
var a = ["1", "2", "3"];

var b = a[1];
a[1] = "4";
// what is the value of b now for all possible data types that the array in "a" might hold?
}

function f() {
var a = [{yellow: "blue"}, {red: "cyan"}, {green: "magenta"}];
var b = a[1];
a[1].red = "tan";
// what is the value of b now and why?

b.red = "black";
// did the value of a[1].red change when I assigned to b.red?
}


If I want to make a fully independent copy of an object (no references whatsoever), what's the best practice way to do that?


Answer



My understanding is that this is actually very simple:





  • Javascript is always pass by value, but when a variable refers to an object (including arrays), the "value" is a reference to the object.

  • Changing the value of a variable never changes the underlying primitive or object, it just points the variable to a new primitive or object.

  • However, changing a property of an object referenced by a variable does change the underlying object.



So, to work through some of your examples:



function f(a,b,c) {
// Argument a is re-assigned to a new value.
// The object or primitive referenced by the original a is unchanged.

a = 3;
// Calling b.push changes its properties - it adds
// a new property b[b.length] with the value "foo".
// So the object referenced by b has been changed.
b.push("foo");
// The "first" property of argument c has been changed.
// So the object referenced by c has been changed (unless c is a primitive)
c.first = false;
}


var x = 4;
var y = ["eeny", "miny", "mo"];
var z = {first: true};
f(x,y,z);
console.log(x, y, z.first); // 4, ["eeny", "miny", "mo", "foo"], false


Example 2:



var a = ["1", "2", {foo:"bar"}];

var b = a[1]; // b is now "2";
var c = a[2]; // c now references {foo:"bar"}
a[1] = "4"; // a is now ["1", "4", {foo:"bar"}]; b still has the value
// it had at the time of assignment
a[2] = "5"; // a is now ["1", "4", "5"]; c still has the value
// it had at the time of assignment, i.e. a reference to
// the object {foo:"bar"}
console.log(b, c.foo); // "2" "bar"

No comments:

Post a Comment

c++ - Does curly brackets matter for empty constructor?

Those brackets declare an empty, inline constructor. In that case, with them, the constructor does exist, it merely does nothing more than t...