Ruby方法Array#<<不更新哈希数组

前端之家收集整理的这篇文章主要介绍了Ruby方法Array#<<不更新哈希数组前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
灵感来自于 How can I marshal a hash with arrays?我不知道Array#<<将无法在以下代码中正常工作:
h = Hash.new{Array.new}
#=> {}
h[0]
#=> []
h[0] << 'a'
#=> ["a"]
h[0]
#=> [] # why?!
h[0] += ['a']
#=> ["a"]
h[0]
#=> ["a"] # as expected

这是否与以下事实有关:<将阵列更改为原位,而Array#创建一个新的实例?

解决方法

如果使用Hash.new的块形式创建哈希,则每次尝试访问实际不存在的元素时,该块都将被执行.所以,我们来看看会发生什么:
h = Hash.new { [] }
h[0] << 'a'

这里得到评估的第一件事就是表达

h[0]

评估时会发生什么?那么块就可以运行了

[]

这不是很令人兴奋:块只是创建一个空数组并返回它.它不做任何其他事情.特别是,它不会以任何方式改变h:h仍然是空的.

接下来,消息<<一个参数'a'被发送到h [0]的结果,这是块的结果,这只是一个空数组:

[] << 'a'

这是做什么的?它将元素’a’添加到空数组,但是由于数组实际上并没有被赋值给任何变量,所以它立即被垃圾收集并消失.

现在,如果再次评估h [0]:

h[0] # => []

h仍然是空的,因为没有分配给它,所以键0仍然不存在,这意味着块再次运行,这意味着它再次返回一个空数组(但是注意到它是一个全新的,不同的空阵列).

h[0] += ['a']

这里发生了什么?首先,运算符分配被取消

h[0] = h[0] + ['a']

现在,右侧的h [0]被评估.还有什么回报?我们已经过去了:h [0]不存在,因此块被运行,块返回一个空数组.再次,这是一个全新的,第三个空的阵列.这个空数组用参数[‘a’]发送消息,这使得它返回另一个数组[‘a’]的新数组.然后将该数组分配给h [0].

最后,在这一点上:

h[0] # => ['a']

现在你终于真的把东西放到了h [0]中,所以很明显你会把你放在的东西.

所以,为了回答你可能遇到的问题,你为什么不知道你放在哪里?你没有把东西放在第一位!

如果你真的要分配到块内的哈希,你必须,很好地分配到块内的哈希:

h = Hash.new {|this_hash,nonexistent_key| this_hash[nonexistent_key] = [] }
h[0] << 'a'
h[0] # => ['a']

如果您查看所涉及的对象的身份,那么在代码示例中看到发生了什么事实真的很容易.那么你可以看到,每当你打电话给h [0],你会得到一个不同的数组.

猜你在找的Ruby相关文章