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

我正在使用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方法都传递了这个参数,它将以相同的方式为两个构造函数使用它。

这是一个常见的误解。 在第一个示例中,您将创建一个包含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] 

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

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

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

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

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

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

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

从ruby文档:

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

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

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