我看到Ruby 1.9有一个新的枚举器slice_before
。 API docs非常神秘。
特别是我对变为initial_state
值的变化感到困惑。
例如,每当元素的渐进总和超过某个值时,我想将带数字的数组拆分为子数组:
a = [1,2,0,1,2,3]
a.slice_before(0) do |elem, sum|
sum += elem
sum > 3
end.to_a
预期产出:
[[1,2,0], [1,2], [3]]
我认为总和就像inject
中的“随身携带”或“备忘录”一样,但似乎并没有成功。
此代码中的故障是一个神秘的错误:
TypeError: can't dup Fixnum
from (irb):43:in `each'
看起来slice_before
不接受Fixnum
作为初始值。为什么? Ruby bug?。
我可以通过保留自己的状态变量来解决这个问题,但这并不是我想要的漂亮的Ruby语义。
sum = 0
a.slice_before do |elem|
sum += elem
sum > 3 && sum = 0
end.to_a
# => [[1, 2, 0], [1, 2], [3]]
initial_state
也可以用于此目的吗?文档中的示例似乎主要是关于文本处理。我正在使用Ruby 1.9.3p194。
答案 0 :(得分:5)
initial_state
通常是存储键值对的状态哈希。
使用状态哈希编写代码:
a.slice_before(sum: 0) do |elem, state|
state[:sum] += elem
if state[:sum] > 3
# Reset the sum, so we'll get more elements
state[:sum] = 0
end
end.to_a
initial_state
必须响应#dup
方法,因为它在每个循环中都是重复的。
Fixnum
不起作用的原因是因为它不响应#dup
。 Fixnum不起作用,因为它无法跟踪每个循环的状态。