迭代器并不是Ruby發明的.它廣泛地運用于各種面向對象語言.在Lisp中也有,只是不這么叫罷了.盡管如此,迭代器的概念并不為許多人熟悉,因此我們將在此做較為詳細的介紹.
你知道,動詞iterate的意思是做同一件事許多遍,因此,iterator就是用來將同一件事做許多次的東西.
當我們寫代碼時,我們需要各種環境下的循環.在C里,我們用for或者while.比如,
char*str;
for(str="abcdefg";*str!='\0';str++){
/*processacharacterhere*/
}
C的for(...)語法提供了一種寫循環的抽象方法,但測試*str是否為空(null)字符需要程序員了解字符串內部結構的細節.這讓C看起來像低級(low-level)語言.更高級的語言是通過它們更具彈性的迭代器支持來實現的.考慮下面的sh命令行腳本:
#!/bin/sh
foriin*.[ch];do
#...herewouldbesomethingtodoforeachfile
done
當前目錄下所有的C源文件和頭文件都將被處理,由命令行shell來一個個地撿取文件名并處理其中的細節.我想這是在比C要高的級別上工作,你覺得呢?
但有更多值得我們考慮的:在一種語言能夠很好的給內建的數據類型的提供迭代器的同時,我們卻仍需要回去用低級別的循環語言來實現對自己定義的數據類型的迭代,這真是讓人失望.在面對對象編程時,用戶經常一個接一個地定義數據類型,因此這是一個很嚴重的問題.
因此,所有的OOP語言都包含了一定的迭代器機制.某些語言為此提供一種特殊的類;Ruby則允許我們直接定義迭代器.
Ruby的String類型有很多有用的迭代器:
ruby>"abc".each_byte{|c|printf"<%c>",c};print"\n"
each_byte是個用于字符串中每個字符的迭代器.每個字符串由局部變量c代替.這可以翻譯為類似C的代碼...
ruby>"a\nb\nc\n".each_line{|l|printl}
采用迭代器,這將很輕松的取代C的大多數編程效果(找換行符,生成子串等等)
前面出現的for語句通過each迭代器實現迭代功能.String的each和each_line的工作原理差不多,讓我們用for重寫上面的例子:
我們可以用retry流程控制語句連接迭代循環,它會從頭執行當前循環的迭代.
yield有時會在一個迭代器的定義中出現.yield將流程控制移至傳遞給迭代器的代碼域(這將會在過程對象那一節介紹更多的細節).下面的例子定義了一個repeat迭代器,會依參數的設置執行多次代碼域.
利用retry,我們可以定義一個有while相同作用的迭代器,雖然在實際應用中它太慢了.