迭代器并不是Ruby發(fā)明的.它廣泛地運(yùn)用于各種面向?qū)ο笳Z(yǔ)言.在Lisp中也有,只是不這么叫罷了.盡管如此,迭代器的概念并不為許多人熟悉,因此我們將在此做較為詳細(xì)的介紹.
你知道,動(dòng)詞 iterate 的意思是做同一件事許多遍,因此,iterator就是用來(lái)將同一件事做許多次的東西.
當(dāng)我們寫(xiě)代碼時(shí),我們需要各種環(huán)境下的循環(huán).在C里,我們用for或者while.比如,
char *str;
for (str = "abcdefg"; *str != '/0'; str++) {
/* process a character here */
}
C的for(...)語(yǔ)法提供了一種寫(xiě)循環(huán)的抽象方法,但測(cè)試 *str 是否為空(null)字符需要程序員了解字符串內(nèi)部結(jié)構(gòu)的細(xì)節(jié).這讓C看起來(lái)像低級(jí)(low-level)語(yǔ)言.更高級(jí)的語(yǔ)言是通過(guò)它們更具彈性的迭代器支持來(lái)實(shí)現(xiàn)的.考慮下面的 sh 命令行腳本:
#!/bin/sh
for i in *.[ch]; do
# ... here would be something to do for each file
done
當(dāng)前目錄下所有的C源文件和頭文件都將被處理,由命令行shell來(lái)一個(gè)個(gè)地?fù)烊∥募⑻幚砥渲械募?xì)節(jié).我想這是在比 C 要高的級(jí)別上工作,你覺(jué)得呢?
但有更多值得我們考慮的:在一種語(yǔ)言能夠很好的給內(nèi)建的數(shù)據(jù)類型的提供迭代器的同時(shí),我們卻仍需要回去用低級(jí)別的循環(huán)語(yǔ)言來(lái)實(shí)現(xiàn)對(duì)自己定義的數(shù)據(jù)類型的迭代,這真是讓人失望.在面對(duì)對(duì)象編程時(shí),用戶經(jīng)常一個(gè)接一個(gè)地定義數(shù)據(jù)類型,因此這是一個(gè)很?chē)?yán)重的問(wèn)題.
因此,所有的OOP語(yǔ)言都包含了一定的迭代器機(jī)制.某些語(yǔ)言為此提供一種特殊的類;Ruby則允許我們直接定義迭代器.
Ruby的String類型有很多有用的迭代器:
ruby> "abc".each_byte{|c| printf "<%c>", c}; print "/n"
<a><b><c>
nil
each_byte 是個(gè)用于字符串中每個(gè)字符的迭代器.每個(gè)字符串由局部變量c代替.這可以翻譯為類似C的代碼...
ruby> s="abc";i=0
0
ruby> while i<s.length
| printf "<%c>", s[i]; i+=1
| end; print "/n"
<a><b><c>
nil
...然而, each_byte 迭代器在概念上要簡(jiǎn)單些,而且即使以后 String 類突然有所改變也應(yīng)該可以照樣工作.使用迭代器的一個(gè)好處便是在此類改變中仍然可以保持健壯;一般的,它的確是好代碼的一個(gè)特點(diǎn).(好,請(qǐng)有點(diǎn)兒耐心,我們將會(huì)馬上談到什么是類)
String的另一個(gè)迭代器是 each_line.