我有一个很大的范围来迭代,找到第一个满足特定约束的元素。
# Runs until memory is exhausted _without_ lazy!
(1..).lazy.select { |i| i > 5 }.first
# => 6
字符串
然而在我的用例中,我想在一个范围的 * 随机间隔 * 开始迭代,如果在到达范围的末尾时没有元素通过检查,则从范围的开始处继续(直到再次到达随机间隔,如果需要的话)。
letter = ('b'..'y').to_a.sample
[*letter..'z', *'a'...letter].map { |c| c.capitalize }.join
# => "FGHIJKLMNOPQRSTUVWXYZABCDE"
型
当然,我没有字母表作为迭代的范围,这只是一个小规模的例子,对于我的用例来说是失败的。
*
(splat)运算符不是懒惰的map
不是懒惰的
通过更多的谷歌搜索和实验,我得到了以下结构:
# lazy version of previous alphabet example
[(letter..'z'), ('a'...letter)].lazy.flat_map { |r| r.each.lazy }.map { |c| c.capitalize }.force.join
=> "FGHIJKLMNOPQRSTUVWXYZABCDE"
# Comparable to what I want
start = rand(2**64)
# => 15282219649142738977
[(start..2**64), (0...start)].lazy.flat_map { |r| r.each.lazy }.select { |i| i % 7 == 0 }.first(5)
# => [15282219649142738978, 15282219649142738985, 15282219649142738992, 15282219649142738999, 15282219649142739006]
iter = [(start..2**64), (0...start)].lazy.flat_map { |r| r.each.lazy }.select { |i| i % 7 == 0 }
# => #<Enumerator::Lazy: #<Enumerator::Lazy: #<Enumerator::Lazy: [15282219649142738977..18446744073709551616, 0...15282219649142738977]>:flat_map>:select>
iter.next
# => 15282219649142738978
iter.next
# => 15282219649142738985
型
在我看来,这确实太复杂了,也许有人有更好的主意?
感谢您抽出宝贵的时间,
泽维尔
2条答案
按热度按时间8zzbczxx1#
如何使用Ruby range?
您可以通过
+
连接枚举器。范围不是枚举器,但您可以通过Range#each
检索枚举器,例如:字符串
合并的枚举器将对每个串联的枚举器进行重新排序:
型
我有一个非常大的范围要遍历,并找到满足特定约束的第一个元素
Ruby有一个专门的方法
Enumerable#find
,它会迭代集合,并返回第一个块返回true结果的元素(不再迭代),例如:型
8ljdwjyq2#
可以使用
Enumerable#chain
字符串