为什么array.each行为依赖于Array.new语法?

时间:2012-01-27 16:29:51

标签: ruby

我正在使用Ruby 1.9.2-p290并找到:

a = Array.new(2, []).each {|i| i.push("a")}    
=> [["a", "a"], ["a", "a"]]

这不是我所期望的。但是下面的构造函数样式确实做了我期望的事情:

b = Array.new(2) {Array.new}.each {|i| i.push("b")}
=> [["b"], ["b"]] 

第一个例子是预期的行为吗?

在ruby-doc中,看起来我的size=2参数与两个构造函数的参数类型相同。我认为如果each方法传递了该参数,它将以相同的方式对两个构造函数使用它。

3 个答案:

答案 0 :(得分:10)

这是一种常见的误解。在第一个示例中,您将创建一个包含2个元素的数组。 这两个都是指向相同数组的指针。因此,当您遍历外部数组时,将2个元素添加到内部数组,然后在输出中反映两次

比较这些:

> array = Array.new(5, [])
=> [[], [], [], [], []] 

# Note - 5 identical object IDs (memory locations)
> array.map { |o| o.object_id }
=> [70228709214620, 70228709214620, 70228709214620, 70228709214620, 70228709214620] 

> array = Array.new(5) { [] }
=> [[], [], [], [], []] 

# Note - 5 different object IDs (memory locations)
> array.map { |o| o.object_id }
=> [70228709185900, 70228709185880, 70228709185860, 70228709185840, 70228709185780] 

答案 1 :(得分:4)

在第一种情况下,您使用Array的单个实例作为主Array元素的默认值:

a = Array.new(2, []).each {|i| i.push("a")}

第二个参数只是循环使用,因此push被应用于同一个实例两次。你只在这里创建了一个实例,一个是作为参数提供的,所以它被反复使用。

第二种方法是正确的方法:

b = Array.new(2) {Array.new}.each {|i| i.push("b")

这故意为主阵列中的每个位置创建一个新的Array实例。这里的重要区别是使用块{ ... },它对新数组中的每个位置执行一次。一个简短形式的版本是:

b = Array.new(2) { [ ] }.each {|i| i.push("b")

答案 2 :(得分:1)

来自ruby文档:

new(size=0, obj=nil)
new(array)
new(size) {|index| block }
  

返回一个新数组。在第一种形式中,新数组为空。在第二个中,它使用obj的大小副本创建(即,大小引用相同的obj)。第三种形式创建作为参数传递的数组的副本(通过在参数上调用to_ary生成数组)。在最后一种形式中,创建了给定大小的数组。

因此,在您创建的a数组中,您有两个对同一数组的引用,因此push对它们都起作用。也就是说,你将"a"推到同一个数组上两次。在您创建的b数组中,您实际上是为每个元素创建一个新数组。