tags:

views:

52

answers:

2

I was trying to write a simple array flatten method, but it does not work using instance variable. It works only using class variables. Can anyone tell me why? and how to make it work using instance variables.

 class Array
 @y = []
  def flatten_array
   self.each do |x|
    if x.class.to_s != 'Array'
    @y <<  x
     else
     x.flatten_array
    end
   end 
   return @y  
  end  
 end


 a =  [1,2,3,4,5]
 b =  [6,7,8]
 c =  [9,10]
 a1 = [12,13,a,b,c]
 puts a1.inspect
 b1 = a1.flatten_array
 puts b1.inspect
+1  A: 

The reason why it doesn't work for instance variables is that x.flatten_array is using a new @y, not the one you were accumulating. Using an instance or class variable this way, however, is not good practice as it is effectively using a global variable for a local function.

The standard procedure for writing a recursive function that needs extra variables is to use a helper function:

def flatten_array
  int_flatten(self, [])
end

private 
def int_flatten(a, result)
  a.each do |elem|
    if (elem.is_a? Array)
      int_flatten(elem, result)
    else
      result << elem
    end
  end
  return result
end

or just be willing to use concatenation:

def flatten_array
  result = []
  each do |elem|
    if (elem.is_a? Array)
      result += elem.flatten_array
    else
      result << elem
    end
  end
  return result
end
Kathy Van Stone
to: Marc-André LafortuneI just wanted to write my own array_flatten to understand Ruby better.to: Andrew GrimmThanks for pointing out the typo in heading. However, that did not answer my question.Kathy,Thanks for the reply.I was thinking that local variable may not work, but apparently I was wrong. I tried both your code snippets. The second worked and first did not. Please see below.
Nick
Snippet1:class Arraydef flatten_array int_flatten(self, [])endprivate def int_flatten(a, result) a.each do |elem| if (elem.is_a? Array) int_flatten(elem, result) else result << elem end endendend a = [1,2,3,[4,5,6,[7,8,9]]] b = a.flatten_array puts b.inspectoutput:[1, 2, 3, [4, 5, 6, [7, 8, 9]]]
Nick
Snippet2:class Arraydef flatten_array result = [] each do |elem| if (elem.is_a? Array) result += elem.flatten_array else result << elem end end return resultendend a = [1,2,3,[4,5,6,[7,8,9]]] b = a.flatten_array puts b.inspectoutput:[1, 2, 3, 4, 5, 6, 7, 8, 9]
Nick
Any idea what wrong with first one?
Nick
Kathy, you missed the return result in the first one.It works after putting in the return..class Arraydef flatten_array result = int_flatten(self, []) return resultendprivate def int_flatten(a, result) a.each do |elem| if (elem.is_a? Array) int_flatten(elem, result) else result << elem end return result endendend a = [1,2,3,[4,5,6,[7,8,9]]] b = a.flatten_array puts b.inspectoutput: [1, 2, 3, 4, 5, 6, 7, 8, 9]
Nick
I'll edit my response. I too saw that I missed a return result.
Kathy Van Stone
A: 

Use native Array#flatten

a =  [1,2,3,4,5]
b =  [6,7,8]
c =  [9,10]
a1 = [12,13,a,b,c]

puts a1.inspect
#=> [12, 13, [1, 2, 3, 4, 5], [6, 7, 8], [9, 10]]

puts a1.flatten.inspect
#=> [12, 13, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
macek