現(xiàn)在我們將前面的一些示例程序的代碼坼開來分析一下.
下面的例子出現(xiàn)在簡單的例子一節(jié).
def?fact(n)??
????if?n?==?0????
???????1??
????else????
???????n?*?fact(n-1)??
????end
end
print?fact(ARGV[0].to_i),?"\n"?
因為是第一次解釋,我們將逐行分析.
def?fact(n)
第一行,def?用于定義一個函數(shù)(或者,更準(zhǔn)確地說,一個方法(method);我們會在稍后的一節(jié)中詳細(xì)討論什么是一個方法).這里,它指明?fact?函數(shù)帶一個參數(shù),也就是?n.
if?n?==?0
if?用來檢查一個條件.當(dāng)條件吻合時,執(zhí)行下面的代碼;否則執(zhí)行跟在else后的代碼.
1?
當(dāng)條件成立時if?的值為?1.
else
如果條件不成立,執(zhí)行從這里到end的代碼.
n?*?fact(n-1)
如果條件不滿足,?if的值會是n乘fact(n-1)的結(jié)果.
end
第一個?end?與?if?語句對應(yīng).
end
第二個?end?與?def?語句對應(yīng).
print?fact(ARGV[0].to_i),?"\n"
這句用由命令行指定的值來調(diào)用fact()函數(shù)并打印結(jié)果.
ARGV是一個包含命令行參數(shù)的數(shù)組.ARGV的成員是字符串,所以我們必須通過to_i轉(zhuǎn)化其為整數(shù).?Ruby并不像Perl那樣自動將字符串轉(zhuǎn)化為整數(shù).
Hmmm...如果向程序賦一個負(fù)值作為參數(shù)會怎樣?你看到這個問題了嗎?你可以修復(fù)它嗎?
Strings
下面我們來檢查在字符串這章中出現(xiàn)的猜謎程序.由于這個要長一點,我們?yōu)槊恳恍写蛏闲袛?shù).
01?words?=?['foobar',?'baz',?'quux']
02?secret?=?words[rand(3)]
03
04?print?"guess??"
05?while?guess?=?STDIN.gets
06???guess.chop!
07???if?guess?==?secret
08?????print?"you?win\n"
09?????break
10???else
11?????print?"you?lose.\n"
12???end
13???print?"guess??"
14?end
15?print?"the?word?is?",?secret,?".\n"
這個程序里,我們使用了一個新的控制結(jié)構(gòu)?while.只要某個指定的條件保持為真,while和它對應(yīng)的end之間的代碼會重復(fù)執(zhí)行.
行2的rand(3)返回一個介于0-2之間的隨機(jī)數(shù).這個隨機(jī)數(shù)用于提取數(shù)組?words?中的一個成員.
在行5我們通過STDIN.gets方法從標(biāo)準(zhǔn)輸入讀取一行.如果讀行遇到時?EOF?(文件結(jié)束),?gets會返回nil.因此,與while相連的代碼會一直執(zhí)行直到它遇到^D(或DOS下的^Z),表示輸入的結(jié)束.
行6的guess.chop!去掉?guess?的最后一個字符;那一定是個換行符.
行15,我們打印出要猜的詞.我們寫的代碼是上向?print?語句傳遞三個參數(shù)(這三個參數(shù)一個接一個地打印),但也可以用一個參數(shù)等效地打印:?將secret寫為?#{secret}以表明將它是一個要取值的變量,而非一個要打印的一般文字:
print?"the?word?is?#{secret}.\n"?
正則表達(dá)式
最后我們來看看正則表達(dá)式一節(jié)的那個程序.
01?st?=?"\033[7m"
02?en?=?"\033[m"
03
04?while?TRUE
05???print?"str>?"
06???STDOUT.flush
07???str?=?gets
08???break?if?not?str
09???str.chop!
10???print?"pat>?"
11???STDOUT.flush
12???re?=?gets
13???break?if?not?re
14???re.chop!
15???str.gsub!?re,?"#{st}\\&#{en}"
16???print?str,?"\n"
17?end
18?print?"\n"
在行4,while的條件被硬設(shè)為?true,因此這好像構(gòu)成了一個無限循環(huán).但我們在行8和行13放置了break語句以跳出循環(huán).這兩個break語句也是?if?修飾辭(if?modifier)的一個例子.一個"if修飾辭"當(dāng)且僅當(dāng)指明條件滿足時執(zhí)行它左邊的語句.
再說說?chop!?(出現(xiàn)在行9和行14).在Ruby里,我們亦可將"!"和"?"附于某些方法名字后面.驚嘆號(!,有時大聲地念作"bang!")暗示某些東西可能具破壞性(destructive),也就是指,某些東西可以改變它所觸及的東西.?chop!直接作用于一個字符串,但不帶!的chop只會產(chǎn)生一個拷貝.下面有這一區(qū)別的演示.
ruby>?s1?=?"forth"
??"forth"
ruby>?s1.chop!???????#?This?changes?s1.
??"fort"
ruby>?s2?=?s1.chop???#?This?puts?a?changed?copy?in?s2,
??"for"
ruby>?s1?????????????#?...?without?disturbing?s1.
??"fort"
以后你還會遇見以問號(?,有時大聲地念做?"huh?")結(jié)束的方法名;這指"斷言"(prediacte)方法,只會返回true或false.
行15應(yīng)給予注意.首先,注意gsub!也是一個破壞函數(shù).它通過替換所有符合?re?模式字符來修改?str(sub指替換,首字母?g?指全局,?比如,替換所有的匹配而不只是第一個匹配).到此為止,還好;但我們用什么來替代文本中的匹配部分呢??在行1和行2里的?st?和?en?被分別定義為表示反轉(zhuǎn)文本顏色(color-inverted)和恢復(fù)正常文本顏色的ANSI碼.?在行15,它們被#{}括起以確保他們被前面定義的那樣解釋(這樣我們才沒看見變量名被打印出來).在這中間是?"\\&".這是個小把戲.因為替換字符串是由雙引號引起的,一對反斜杠會被解釋為一個單一的反斜杠;所以?gsub!實際上得到的是"\&",就一段特殊代碼正好就是表示"任何與模式于第一處匹配的字符".因此新的字符串在被打印出來的時候,很像原來的那個,只不過匹配的部分以反視(inverse?video)的方式高亮度顯示出來.
版權(quán)聲明:RUBY文檔中心的所有文章標(biāo)明[原創(chuàng)]的均為本站作品,版權(quán)屬RUBY中文化計劃,若轉(zhuǎn)載請注明;標(biāo)明[翻譯]的其外文版權(quán)歸原作者,譯文版權(quán)屬RUBY中文化計劃;標(biāo)明[轉(zhuǎn)貼]的,若原作者感到侵犯了他的著作權(quán),那么請及時跟主持人聯(lián)系,我們會盡快更正。
?