2012-05-23 23 views
5

Sto cercando un metodo che appiattisca un hash "json" in un hash appiattito ma mantieni le informazioni sul percorso nei tasti appiattiti. Per esempio:Appiattisci un oggetto json nidificato

h = {"a" => "foo", "b" => [{"c" => "bar", "d" => ["baz"]}]} 

Flatten (h) dovrebbe restituire:

{"a" => "foo", "b_0_c" => "bar", "b_0_d_0" => "baz"} 
+0

Non penso che "cosa hai provato" si applichi davvero in questa situazione. È una domanda generale e utile da avere nella base di conoscenza. – pguardiario

+0

Forse, ma se l'ho implementato personalmente, non sarebbe qui. Ho pensato che la risposta dovrebbe essere qui, quindi ho postato la domanda. – pguardiario

+0

Oh, sei tu, l'OP. Scusa :) –

risposta

12

Questo dovrebbe risolvere il problema:

h = {'a' => 'foo', 'b' => [{'c' => 'bar', 'd' => ['baz']}]} 

module Enumerable 
    def flatten_with_path(parent_prefix = nil) 
    res = {} 

    self.each_with_index do |elem, i| 
     if elem.is_a?(Array) 
     k, v = elem 
     else 
     k, v = i, elem 
     end 

     key = parent_prefix ? "#{parent_prefix}.#{k}" : k # assign key name for result hash 

     if v.is_a? Enumerable 
     res.merge!(v.flatten_with_path(key)) # recursive call to flatten child elements 
     else 
     res[key] = v 
     end 
    end 

    res 
    end 
end 

puts h.flatten_with_path.inspect 
+0

Soluzione molto bella. – pguardiario

Problemi correlati