Dig 方法¶ ↑
Ruby 的 dig
方法对于访问嵌套数据结构很有用。
考虑以下数据
item = { id: "0001", type: "donut", name: "Cake", ppu: 0.55, batters: { batter: [ {id: "1001", type: "Regular"}, {id: "1002", type: "Chocolate"}, {id: "1003", type: "Blueberry"}, {id: "1004", type: "Devil's Food"} ] }, topping: [ {id: "5001", type: "None"}, {id: "5002", type: "Glazed"}, {id: "5005", type: "Sugar"}, {id: "5007", type: "Powdered Sugar"}, {id: "5006", type: "Chocolate with Sprinkles"}, {id: "5003", type: "Chocolate"}, {id: "5004", type: "Maple"} ] }
如果没有 dig
方法,你可以写
item[:batters][:batter][1][:type] # => "Chocolate"
有了 dig
方法,你可以写
item.dig(:batters, :batter, 1, :type) # => "Chocolate"
如果没有 dig
方法,你可以写(错误地,引发 NoMethodError (undefined method `[]' for nil:NilClass)
)
item[:batters][:BATTER][1][:type]
有了 dig
方法,你可以写(仍然是错误的,但避免了异常)
item.dig(:batters, :BATTER, 1, :type) # => nil
为什么 dig
更好?¶ ↑
-
它具有更少的语法元素(不容易出错)。
-
它读起来更好。
-
如果找不到项目,它不会引发异常。
dig
如何工作?¶ ↑
调用序列是
obj.dig(*identifiers)
identifiers
定义了进入嵌套数据结构的“路径”
-
对于
identifiers
中的每个标识符,使用该标识符在接收器上调用 #dig 方法。 -
第一个接收器是
self
。 -
每个后续接收器都是之前调用
dig
返回的值。 -
最终返回的值是最后一次调用
dig
返回的值。
如果任何接收器不响应 #dig,dig
方法将引发异常
h = { foo: 1 } # Raises TypeError (Integer does not have #dig method): h.dig(:foo, :bar)
还有什么?¶ ↑
上面的结构有 Hash 对象和 Array 对象,它们都有实例方法 dig
。
总共有六个内置的 Ruby 类具有方法 dig
,核心类中有三个,标准库中有三个。
在核心中
-
Array#dig
:第一个参数是一个整数索引。 -
Hash#dig
:第一个参数是一个键。 -
Struct#dig
:第一个参数是一个键。
在标准库中
-
OpenStruct#dig:第一个参数是一个字符串名称。
-
CSV::Table#dig:第一个参数是一个整数索引或一个字符串标题。
-
CSV::Row#dig:第一个参数是一个整数索引或一个字符串标题。