迭代时删除数组元素

Abh*_*hek 7 ruby

我正在使用两个each块迭代嵌套数组,并从内部迭代内的同一个数组中删除一个元素:

arr = [1,2,3]  
arr.each do |x|  
  arr.each do |y|  
    puts "#{arr.delete(y)}"  
  end  
end
Run Code Online (Sandbox Code Playgroud)

这将产生的结果1,3.阵列变成了[2].

为什么值不2传递给第一个或第二个循环?这是嵌套迭代的某种副作用吗?

gui*_*man 9

这是因为已删除元素的索引.我添加了一些输出来告诉你:

arr = [1,2,3]  
arr.each do |x|  
  puts "1: #{arr.inspect}, x: #{x}"
  arr.each do |y|  
    puts "2: #{arr.inspect}, y: #{y}"
    puts "#{arr.delete(y)}"  
  end  
end
Run Code Online (Sandbox Code Playgroud)

结果:

1: [1, 2, 3], x: 1
2: [1, 2, 3], y: 1
1
2: [2, 3], y: 3
3
=> [2]
Run Code Online (Sandbox Code Playgroud)

每个块内部的第一个删除元素是1(索引为0).在删除2之后,索引为0,现在每次迭代都转到索引1,现在是元素3. 3将被删除,这就是迭代的结束.所以你得到了[2].

没有嵌套的每个都会发生同样的事情

arr = [1,2,3]  
arr.each do |x|  
  puts "1: #{arr.inspect}, x: #{x}" 
  puts "#{arr.delete(x)}"  
end
Run Code Online (Sandbox Code Playgroud)

结果:

1: [1, 2, 3], x: 1
1
1: [2, 3], x: 3
3
=> [2]
Run Code Online (Sandbox Code Playgroud)

我建议使用reverse_each此类操作来避免此行为:

arr = [1,2,3]  
arr.reverse_each do |x|  
  puts "1: #{arr.inspect}, x: #{x}" 
  puts "#{arr.delete(x)}"  
end
Run Code Online (Sandbox Code Playgroud)

结果:

1: [1, 2, 3], x: 3
3
1: [1, 2], x: 2
2
1: [1], x: 1
1
=> []
Run Code Online (Sandbox Code Playgroud)