如何生成包含散列和数组作为值的嵌套散列的直接访问密钥?

时间:2015-05-15 13:17:46

标签: ruby

我想比较两个XML文件,其中一个是输入,另一个是输出。我将两者都转换为哈希。

我的想法是将输入XML中的所有键转换为散列,并在输入和输出散列中搜索各自的键/值对中的每个键。

我有一个哈希:

{
  "requisition_header" => {
    "requested_by" => {"login" => "coupasupport"},
    "department" => {"name" => "Marketing"},
    "ship_to_address" => {"name" => "Address_1431693296"},
    "justification" => nil,
    "attachments" => [],
    "requisition_lines" => [
      {
        "description" => "Cleaning Services for Building A",
        "line_num" => 1,
        "need_by_date" => 2010-09-23 07:00:00 UTC,
        "source_part_num" => nil,
        "supp_aux_part_num" => nil,
        "unit_price" => #<BigDecimal:a60520c,'0.3E4',9(18)>,
        "supplier" => {"name" => "amazon.com"},
          "account" => {
          "code" => "SF-Marketing-Indirect",
          "account_type" => {"name" => "Ace Corporate"}
        },
        "currency" => {"code" => "USD"},
        "payment_term" => {"code" => "Net 30"},
        "shipping_term" => {"code" => "Standard"},
        "commodity" => {"name" => "Marketing-Services"}
      }
    ]
  }
}

它是嵌套的,并且无法直接访问所有值。

我想要一种方法来生成对哈希值中每个值的直接访问。

例如:

requisition_header.requested_by.login

将访问“coupasupport”。

requisition_header.department.name

将访问“营销”。

requisition_header.requisition_lines[0].description

将访问“A楼的清洁服务”。

requisition_header.requisition_lines[0].line_num

将访问“1”。

requisition_header.requisition_lines[0].need_by_date

将访问“2010-09-23 07:00:00 UTC”。

构建的每个密钥都可用于直接在哈希中搜索值。

3 个答案:

答案 0 :(得分:3)

可以使用以下方法完成,将嵌套哈希转换为嵌套OpenStruct

require 'ostruct'
def deep_structify(hash)
  result = {}
  hash.each do |key, value|
    result[key] = value.is_a?(Hash) ? deep_structify(value) : value
  end if hash
  OpenStruct.new(result)
end

hash = {"requisition_header"=>{"requested_by"=>{"login"=>"coupasupport"}, "department"=>{"name"=>"Marketing"}, "ship_to_address"=>{"name"=>"Address_1431693296"}, "justification"=>nil, "attachments"=>[], "requisition_lines"=>[{"description"=>"Cleaning Services for Building A", "line_num"=>1, "need_by_date"=>2010-09-23 07:00:00 UTC, "source_part_num"=>nil, "supp_aux_part_num"=>nil, "unit_price"=>#<BigDecimal:a60520c,'0.3E4',9(18)>, "supplier"=>{"name"=>"amazon.com"}, "account"=>{"code"=>"SF-Marketing-Indirect", "account_type"=>{"name"=>"Ace Corporate"}}, "currency"=>{"code"=>"USD"}, "payment_term"=>{"code"=>"Net 30"}, "shipping_term"=>{"code"=>"Standard"}, "commodity"=>{"name"=>"Marketing-Services"}}]}}

struct = deep_structify(hash)

struct.requisition_header.department.name
#=> "Marketing"

答案 1 :(得分:0)

您也可以通过覆盖OpenStruct#new来实现,

require 'ostruct'

class DeepStruct < OpenStruct
  def initialize(hash=nil)
    @table = {}
    @hash_table = {}

    if hash
      hash.each do |k,v|
        @table[k.to_sym] = (v.is_a?(Hash) ? self.class.new(v) : v)
        @hash_table[k.to_sym] = v

        new_ostruct_member(k)
      end
    end
  end

  def to_h
    @hash_table
  end

end

现在你可以做到:

require 'deep_struct'

hash = {"requisition_header"=>{"requested_by"=>{"login"=>"coupasupport"}, "department"=>{"name"=>"Marketing"}, "ship_to_address"=>{"name"=>"Address_1431693296"}, "justification"=>nil, "attachments"=>[], "requisition_lines"=>[{"description"=>"Cleaning Services for Building A", "line_num"=>1, "need_by_date"=>2010-09-23 07:00:00 UTC, "source_part_num"=>nil, "supp_aux_part_num"=>nil, "unit_price"=>#<BigDecimal:a60520c,'0.3E4',9(18)>, "supplier"=>{"name"=>"amazon.com"}, "account"=>{"code"=>"SF-Marketing-Indirect", "account_type"=>{"name"=>"Ace Corporate"}}, "currency"=>{"code"=>"USD"}, "payment_term"=>{"code"=>"Net 30"}, "shipping_term"=>{"code"=>"Standard"}, "commodity"=>{"name"=>"Marketing-Services"}}]}}

mystruct = DeepStruct.new hash
mystruct.requisition_header.requested_by.login # => coupasupport
mystruct.requisition_header.to_h # => {"requested_by"=>{"login"=>"coupasupport"}

答案 2 :(得分:0)

您可以使用BasicObject#method_missing

<强>代码

hash = { animals: {
           pets: { dog: "Diva", cat: "Boots", python: "Stretch" },
           farm: { pig: "Porky", chicken: "Little", sheep: "Baa" }
         },
         finishes: {
           tinted: { stain: "Millers", paint: "Oxford" },  
           clear:  { lacquer: "Target", varnish: "Topcoat" }
         }
       }

hash.finishes.tinted.stain
  #=> "Millers
hash.animals.pets.cat
  #=> "Boots"
hash.animals.pets
  #=> {:dog=>"Diva", :cat=>"Boots", :python=>"Stretch"}
hash.animals
  #=> {:pets=>{:dog=>"Diva", :cat=>"Boots", :python=>"Stretch"},
  #    :farm=>{:pig=>"Porky", :chicken=>"Little", :sheep=>"Baa"}} 

示例

{{1}}

读者挑战

这种方法存在潜在的“陷阱”。我留给读者来识别它。我的例子包含一个线索。 (请注意,可能还有其他一些我没有想过的问题。)