ruby 中的递归函数正在覆盖克隆的 (object.dup) 变量的嵌套属性

Recursive function in ruby is overwriting nested attributes of cloned(object.dup) variable

我有这样的哈希:

entity = {1=> nil, 2 => {3 => nil, 4 => 1}}

我写了一个函数,它可以使用递归删除给定实体的空值。

def clear_null_values(entity)
   entity.each do |key, value|
     if value == nil || value.blank?
       entity.delete(key)
     elsif value.is_a? Hash
       clear_null_values(value)
       entity.delete(key) if value.blank?
     end
   end
end 

而且我还需要原始实体用于其他目的。所以我复制了散列变量,然后清除了空值。

final_entity = entity.dup
clear_null_values(entity)
puts entity
puts final_entity

结果:

{2 => {4 => 1}}
{1=> nil, 2 => {4 => 1}} # the nested values are overwritten.

理想情况下,final_entity 应与原始实体相同。

问题 1:为什么 entity.dup 只复制 outerhash?

问题 2:如何制作 final_entity 原始实体的精确副本,即即使我们修改了实体,那么 final_entity 也不应该改变?

尝试使用 deep_dup,您的原始代码仅 dup-ed 最外层的哈希值。

final_entity = entity.deep_dup
clear_null_values(entity)
puts entity
puts final_entity

输出:

{2=>{4=>1}}
{1=>nil, 2=>{3=>nil, 4=>1}}

注意:Rails 还添加了 Hash#compact,您可以使用它来简化 clear_null_values

在我看来,通过直接在 entities 上而不是在它的副本上操作来计算去除 nil 值的散列会更清晰。

def clear_null_values(entity)
  entity.each_with_object({}) do |(k,v),h|
    next if v.nil?
    h[k] = Hash === v ? clear_null_values(v) : v
  end
end

entities = { 1=>nil, 2=>{ 3=>nil, 4=>1 } }

clear_null_values entities
  #=> {2=>{4=>1}}

我们可以确认 entities 没有发生突变。

entities
  #=> {1=>nil, 2=>{3=>nil, 4=>1}}