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