递归地将Hash转换为OpenStruct

Don*_*ato 14 ruby

鉴于我有这个哈希:

 h = { a: 'a', b: 'b', c: { d: 'd', e: 'e'} }
Run Code Online (Sandbox Code Playgroud)

我转换为OpenStruct:

o = OpenStruct.new(h)
 => #<OpenStruct a="a", b="b", c={:d=>"d", :e=>"e"}> 
o.a
 => "a" 
o.b
 => "b" 
o.c
 => {:d=>"d", :e=>"e"} 
2.1.2 :006 > o.c.d
NoMethodError: undefined method `d' for {:d=>"d", :e=>"e"}:Hash
Run Code Online (Sandbox Code Playgroud)

我希望所有嵌套键都是方法.所以我可以这样访问d:

o.c.d
=> "d"
Run Code Online (Sandbox Code Playgroud)

我怎样才能做到这一点?

Cru*_*nez 26

你可以修补Hash课程

class Hash
  def to_o
    JSON.parse to_json, object_class: OpenStruct
  end
end
Run Code Online (Sandbox Code Playgroud)

那么你可以说

h = { a: 'a', b: 'b', c: { d: 'd', e: 'e'} }
o = h.to_o
o.c.d # => 'd'
Run Code Online (Sandbox Code Playgroud)

请参阅将复杂的嵌套哈希转换为对象.

  • 使用Ruby已有10多年了,我仍然对它的优雅感到惊讶。 (4认同)
  • 虽然它应该可以解决问题,但是这个过程有一些副作用,比如它会改变项目中所有地方的默认 `Hash` 行为,这可能会在未来几天产生不可预见的问题。所以我宁愿使用 max_pleaner 或 Donato `JSON.parse(h.to_json, object_class: OpenStruct)`,这似乎可以解决问题而没有副作用。 (2认同)
  • 仅供参考,这仅适用于简单的值.当我将此方法应用于更复杂的对象(例如BigDecimal或自定义ActiveRecord对象)时,我遇到了问题 (2认同)

Don*_*ato 15

我想出了这个解决方案:

h = { a: 'a', b: 'b', c: { d: 'd', e: 'e'} }
json = h.to_json
=> "{\"a\":\"a\",\"b\":\"b\",\"c\":{\"d\":\"d\",\"e\":\"e\"}}" 
object = JSON.parse(json, object_class:OpenStruct)
object.c.d
 => "d" 
Run Code Online (Sandbox Code Playgroud)

所以为了这个工作,我不得不做一个额外的步骤:将它转换为json.


max*_*ner 13

我个人使用recursive-open-struct宝石 - 然后就这么简单RecursiveOpenStruct.new(<nested_hash>)

但是为了递归练习,我会告诉你一个新的解决方案:

require 'ostruct'

def to_recursive_ostruct(hash)
  OpenStruct.new(hash.each_with_object({}) do |(key, val), memo|
    memo[key] = val.is_a?(Hash) ? to_recursive_ostruct(val) : val
  end)
end

puts to_recursive_ostruct(a: { b: 1}).a.b
# => 1
Run Code Online (Sandbox Code Playgroud)

编辑

这比基于JSON的解决方案更好的原因是因为转换为JSON时可能会丢失一些数据.例如,如果将Time对象转换为JSON然后解析它,它将是一个字符串.还有很多其他的例子:

class Foo; end
JSON.parse({obj: Foo.new}.to_json)["obj"]
# => "#<Foo:0x00007fc8720198b0>"
Run Code Online (Sandbox Code Playgroud)

是的...不是超级有用的.你已经完全失去了对实际实例的引用.


Xav*_*avi 7

这是一个避免将哈希转换为 json 的递归解决方案:

def to_o(obj)
  if obj.is_a?(Hash)
    return OpenStruct.new(obj.map{ |key, val| [ key, to_o(val) ] }.to_h)
  elsif obj.is_a?(Array)
    return obj.map{ |o| to_o(o) }
  else # Assumed to be a primitive value
    return obj
  end
end
Run Code Online (Sandbox Code Playgroud)