Creating matrix with `Array.new(n, Array.new)`

I created an array by doing the following:

@gameboard = Array.new(3, Array.new(3, " "))

I tried to assign a value like so, and I got this:

@gameboard[0][2] = "X"
@gameboard #=> [[" ", " ", "X"], [" ", " ", "X"], [" ", " ", "X"]]

When I declare the array differently,

@gameboard = [[" ", " ", " "], [" ", " ", " "], [" ", " ", " "]]

I get this result:

@gameboard[0][2] = "X"
@gameboard # => [[" ", " ", "X"], [" ", " ", " "], [" ", " ", " "]]

Why does using the Array.new method illicit different behavior when assigning values to the array?


Solution 1:

Follow the code:

@gameboard = Array.new(3, Array.new(3, " "))
@gameboard.map { |a| a.object_id }
# => [76584030, 76584030, 76584030]

means new(size=0, obj=nil) method creates an array of size, having the same ob.

But new(size) {|index| block } method works in a different way; it creates an array of size, having different obs.

See the code below:

@gameboard = Array.new(3) { Array.new(3, " ") }
@gameboard.map { |a| a.object_id }
# => [75510080, 75509920, 75509540]

The above is the same as your second code example:

@gameboard = [[" ", " ", " "], [" ", " ", " "], [" ", " ", " "]]
@gameboard.map { |a| a.object_id }
# => [80194090, 80193400, 80193080]

If you change or update the the value at index 1 of the first element array of @gameboard, it wouldn't affect all other inner array elements.

@gameboard = Array.new(3) { Array.new(3, " ") }
@gameboard[0][1] = 2
@gameboard
# => [[" ", 2, " "], [" ", " ", " "], [" ", " ", " "]]

Solution 2:

The Array constructor will not duplicate the object you passed; it will reuse the object to fill the array.

Use the block form in order to create a new object for each index:

@gameboard = Array.new(3) { |i| Array.new(3) { |j| " " } }