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],你会得到一个不同的数组. (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |
- c# – 记录文件名,日志中的行号.我需要传递StackFrame吗?
- c – 有效地将std :: unordered_set的内容移动到std :: vec
- PostgreSQL数据库bat文件的制作
- ruby-on-rails – 为什么我收到警告:跳过密钥错误?
- oracle 配置 ACL 使用数据库发送WebServic请求时需要
- react-native – 卸载ReactNative App – 使用Asyncstorage
- ruby-on-rails – 在Rails应用程序中实现投票的最佳方式?
- Flutter 使用Navigator进行局部跳转页面
- 忘记思科交换机密码处理步骤
- 【转】Flex 中用类名反射出类实例(getDefinitionByName)