惰性枚举直到块为假

Enumerate lazily until the block is false

我已经编写了一个 Enumerable class 来无缝地延迟获取 API 请求的所有页面。

class Pager
  include Enumerable

  def initialize(&fetch_next_page)
    @fetch_next_page = fetch_next_page
    reset
  end

  def reset
    @page_number = 0
  end

  private def next_page
    @page_number += 1

    return @fetch_next_page.call(@page_number)
  end

  def each(&block)
    if block_given?
      while resp = next_page
        resp.each(&block)
      end
      reset
    else
      to_enum(:each)
    end
  end
end

这是一个如何使用它的例子。

pager = Pager.new do |page_number|
  response = fetch_page( page: page_number, **some_options )

  response.page <= response.total_pages ? response.stuff : false
end

但我开始意识到所有这一切都在执行一个块,该块 returns 一个 Enumerable 直到它为 false,并且它正在展平 Enumerables。

pager = Pager.new { |page_number|
  page_number <= 3 ? 1.upto(page_number) : false
}

# [1, 1, 2, 1, 2, 3]
puts pager.to_a.inspect

有更简单的方法吗?我已经接近 Enumerator,但无法使扁平化工作。

def paginate(&fetch_next)
  return Enumerator.new do |yielder|
    page_number = 1
    while ret = fetch_next.call(page_number)
      yielder.yield(*ret)
      page_number += 1
    end
  end
end

pager = paginate { |page_number|
  page_number <= 3 ? 1.upto(page_number) : false
}

# [1, [1, 2], [1, 2, 3]]
puts pager.to_a.inspect

枚举器输出不正确的原因确实与 splat 运算符有关。

如果您将多个值传递给 yield 它们会一次性全部生成,而您希望一个一个地生成它们。既然你已经得到了块:

{ |page_number| page_number <= 3 ? 1.upto(page_number) : false }

这将产生 3 个产量。第一个参数 1,第二个参数 1, 2,第三个参数 1, 2, 3。如果您想将它们作为单独的收益来产生,您必须更改以下内容:

yielder.yield(*ret)

# should be changed to

ret.each { |e| yielder.yield e }
# or
ret.each { |e| yielder << e }
# depending on your preference

pager.to_a
#=> [1, 1, 2, 1, 2, 3]