是否可以手动获取“yield”值而不是循环获取?

Is it possible to manually fetch `yield` values instead of in a loop?

以文档中的这个例子为例,我们可以发送一个从函数接收 yield 值的块。

def twice
  yield 1
  yield 2
end

twice do |i|
  puts i
end

我想知道的是我们是否可以手动调用该函数,以便第一次调用给出 1 然后我们做其他事情,然后下一次调用给出 2.

我尝试了(只是猜测)各种方法,例如 puts twice() 但无济于事。有什么办法可以做到吗?

您可以使用出生点和通道。这是您可以获得的最接近的值:

def foo
  yield 1
  yield 2
end

chan = Channel(typeof(foo { |x| x })).new

spawn do
  foo do |x|
    chan.send x
  end
  chan.close
end

p chan.receive?
p chan.receive?
p chan.receive?

http://play.crystal-lang.org/#/r/ijd

你可以在上面建立一个抽象,也许用宏:

def foo
  yield 1
  yield 2
end

macro enumerator(call)
  %chan = Channel(typeof({{call}} { |x| x })).new
  spawn do
    {{call}} do |x|
      %chan.send x
    end
    %chan.close
  end
  %chan
end

values = enumerator(foo)

p values.receive?
p values.receive?
p values.receive?

http://play.crystal-lang.org/#/r/ije

并且可能使用包装器结构,因此它看起来像迭代器 (http://crystal-lang.org/api/Iterator.html)。但请注意,这有点慢,至少比使用迭代器或常规非捕获块调用慢得多。