現在我們將前面的一些樣本程式的代碼坼開來分析一下.
下面的例子出現在簡單的例子一節.
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"
Regex
最後我們來看看Regex一節的那個程式.
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,它們被#{}括起以確保他們被前面定義的那樣解釋(這樣我們才沒看見變數名被列印出來).在這中間是 "\\&".這是個小把戲.因為替換字串是由雙引號引起的,一對反斜線會被解釋為一個單一的反斜線;所以 gsub!實際上得到的是"\&",就一段特殊代碼正好就是表示"任何與模式於第一處匹配的字元".因此新的字串在被列印出來的時候,很像原來的那個,只不過匹配的部分以反視(inverse video)的方式高亮度顯示出來.