Warning: file_get_contents(/data/phpspider/zhask/data//catemap/5/ruby/24.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Ruby 为什么array.each行为依赖于array.new语法?_Ruby - Fatal编程技术网

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

Ruby 为什么array.each行为依赖于array.new语法?,ruby,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文档中,我的size=2参数对于两个构造函数来

我正在使用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文档中,我的
size=2
参数对于两个构造函数来说都是相同类型的参数。我认为如果
每个
方法都通过了该参数,那么它将以相同的方式对两个构造函数使用它

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

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

a = Array.new(2, []).each {|i| i.push("a")}
第二个参数只是循环使用,因此
push
应用于同一实例两次。您在这里只创建了一个实例,一个作为参数提供,因此它被反复使用

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

b = Array.new(2) {Array.new}.each {|i| i.push("b")
这有意为主数组中的每个位置创建一个新的数组实例。这里的重要区别是使用了块
{…}
,它对新数组中的每个位置执行一次。这方面的一个简短版本是:

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

从ruby文档中:

new(size=0, obj=nil)
new(array)
new(size) {|index| block }
返回一个新数组。在第一种形式中,新数组为空。在第二种情况下,它是使用obj的大小副本创建的(即,对同一obj的大小引用)。第三种形式是创建作为参数传递的数组的副本(该数组是通过调用该参数生成的)。在最后一种形式中,将创建给定大小的数组

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