現(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ù)(或者,更準確地說,一個方法(method);我們會在稍后的一節(jié)中詳細討論什么是一個方法).這里,它指明?fact?函數(shù)帶一個參數(shù),也就是?n.
if?n?==?0?
if?用來檢查一個條件.當條件吻合時,執(zhí)行下面的代碼;否則執(zhí)行跟在else后的代碼.
1??
當條件成立時if?的值為?1.
else?
如果條件不成立,執(zhí)行從這里到end的代碼.
n?*?fact(n-1)?
如果條件不滿足,?if的值會是n乘fact(n-1)的結果.
end?
第一個?end?與?if?語句對應.
end?
第二個?end?與?def?語句對應.
print?fact(ARGV[0].to_i),?"/n"?
這句用由命令行指定的值來調用fact()函數(shù)并打印結果.
ARGV是一個包含命令行參數(shù)的數(shù)組.ARGV的成員是字符串,所以我們必須通過to_i轉化其為整數(shù).?Ruby并不像Perl那樣自動將字符串轉化為整數(shù).
Hmmm...如果向程序賦一個負值作為參數(shù)會怎樣?你看到這個問題了嗎?你可以修復它嗎?
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"?
這個程序里,我們使用了一個新的控制結構?while.只要某個指定的條件保持為真,while和它對應的end之間的代碼會重復執(zhí)行.
行2的rand(3)返回一個介于0-2之間的隨機數(shù).這個隨機數(shù)用于提取數(shù)組?words?中的一個成員.
在行5我們通過STDIN.gets方法從標準輸入讀取一行.如果讀行遇到時?EOF?(文件結束),?gets會返回nil.因此,與while相連的代碼會一直執(zhí)行直到它遇到^D(或DOS下的^Z),表示輸入的結束.
行6的guess.chop!去掉?guess?的最后一個字符;那一定是個換行符.
行15,我們打印出要猜的詞.我們寫的代碼是上向?print?語句傳遞三個參數(shù)(這三個參數(shù)一個接一個地打印),但也可以用一個參數(shù)等效地打印:?將secret寫為?#{secret}以表明將它是一個要取值的變量,而非一個要打印的一般文字:
print?"the?word?is?#{secret}./n"??
正則表達式
最后我們來看看正則表達式一節(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的條件被硬設為?true,因此這好像構成了一個無限循環(huán).但我們在行8和行13放置了break語句以跳出循環(huán).這兩個break語句也是?if?修飾辭(if?modifier)的一個例子.一個"if修飾辭"當且僅當指明條件滿足時執(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?")結束的方法名;這指"斷言"(prediacte)方法,只會返回true或false.
行15應給予注意.首先,注意gsub!也是一個破壞函數(shù).它通過替換所有符合?re?模式字符來修改?str(sub指替換,首字母?g?指全局,?比如,替換所有的匹配而不只是第一個匹配).到此為止,還好;但我們用什么來替代文本中的匹配部分呢??在行1和行2里的?st?和?en?被分別定義為表示反轉文本顏色(color-inverted)和恢復正常文本顏色的ANSI碼.?在行15,它們被#{}括起以確保他們被前面定義的那樣解釋(這樣我們才沒看見變量名被打印出來).在這中間是?"http://&".這是個小把戲.因為替換字符串是由雙引號引起的,一對反斜杠會被解釋為一個單一的反斜杠;所以?gsub!實際上得到的是"/&",就一段特殊代碼正好就是表示"任何與模式于第一處匹配的字符".因此新的字符串在被打印出來的時候,很像原來的那個,只不過匹配的部分以反視(inverse?video)的方式高亮度顯示出來.