文章1-4篇说的都是js中的可迭代对象,下面让我们看看ruby中的等价物。
不可否认,ruby中对于迭代器和生成器的语法都相当简洁;ruby从一开始就有一个简洁的基因,而js后来的不断扩充使得其有些语法比较“别扭”和“奇怪”,虽说ruby也不比js小几岁啊!(官方的说法是一个1992年一个1995年)
在ruby中遍历一个数组的代码如下所示:
而在ruby中范围对象本身是可枚举的(可遍历差不多意思),所以你可以直接这么写:
在ruby中写一个迭代器,也是相当简单:
而ruby中的枚举器和js中的可迭代对象类似,枚举器是类enumerable::enumerator的实例,一般通过object类的to_enum或其同义词enum_for方法来创建连如果调用时没有提供参数则to_enum返回一个枚举器,该枚举器的each方法只是简单调用目标对象的each方法,枚举器的一个用途是建立防御式拷贝:
如果to_enum中第一个实参是一个符号,则它标识了一个迭代器方法,枚举器的each方法调用那个迭代器方法:
而在ruby1.9之后又对枚举器的语法做了修改使其更加简洁,当以不带代码块的方式调用内建迭代器方法时,会自动返回一个枚举器,因此上述代码可以简写为:
要想对自定义的迭代器方法实现如上行为,可以通过返回self.to_enum来实现:
简单解释下self.to_enum(:itor,x)这句,因为我为了测试方便是在全局上下文中定义的函数itor,所以它自动成为全局对象main的私有方法,而该方法也会成为object类的私有实例方法:
在执行self.to_enum(:itor,x)时,当时的self是main对象,即将返回枚举器的each方法与itor函数对应上。
ruby里面也有类似的for in循环,不过我们一般都不怎么用呢:
枚举器可以实现外部迭代,实际上枚举器也称为外部迭代器。可以使用kernel.loop方法包裹枚举器,从而在stopiteration异常抛出时自动退出循环:
在ruby中枚举器自带了rewind方法使外部迭代器重新开始迭代。
最后,在ruby中我们可以用纤程(fiber)来实现所谓的生成器,虽然这比较晦涩: