前面,我們說Ruby沒有函數(shù),只有方法.而且實(shí)際上有不止一種方法.這一節(jié)我們介紹訪問控制(accesscontrols).
想想當(dāng)我們?cè)?quot;最高層"而不是在一個(gè)類的定義里定義一個(gè)方法時(shí)會(huì)發(fā)生什么.我們可以把這樣的方法設(shè)想為在像C那樣的傳統(tǒng)語言中的函數(shù)的類似物.
ruby>defsquare(n)
|n*n
|end
nil
ruby>square(5)
25
我們的新方法看起來不屬于任何類,但實(shí)際上Ruby將其分給Object類,也就是所有其它類的父類.因此,所有對(duì)象現(xiàn)在都可以使用這一方法.這本應(yīng)是正確的,但有個(gè)小陷阱:它是所有類的私有(private)方法.我們將在下面討論這是什么意思,但一個(gè)結(jié)果是它只能以函數(shù)的風(fēng)格調(diào)用,像這樣:
ruby>classFoo
|deffourth_power_of(x)
|square(x)*square(x)
|end
|end
nil
ruby>Foo.new.fourth_power_of10
10000
我們不允許向一個(gè)對(duì)象明確地運(yùn)用這一方法:
ruby>"fish".square(5)
ERR:(eval):1:privatemethod`square'calledfor"fish":String
這一聰明的做法在ruby使得提供可以像在傳統(tǒng)語言中那樣運(yùn)用函數(shù)的同時(shí)保持了ruby的純OO性質(zhì)(函數(shù)仍是對(duì)象方法,但接受者隱式的為self).
在OO編程里,有一個(gè)不成文的習(xí)慣,我們?cè)谇懊娴恼鹿?jié)里有所暗示的,即有關(guān)規(guī)格(specification)和實(shí)現(xiàn)(implementation)的區(qū)別,或者說對(duì)象被要求完成什么任務(wù)和實(shí)際上它是怎樣完成的.對(duì)象的內(nèi)部工作應(yīng)該對(duì)用戶保持隱蔽;他們應(yīng)該只關(guān)心輸入什么和輸出什么,并相信對(duì)象知道它在內(nèi)部是在做什么.如此,某些外面世界看不見但卻在內(nèi)部使用(并可以由程序員在任何需要的時(shí)候改進(jìn),而不用改變用戶看到的類的對(duì)象)的方法將很有用.在下面這個(gè)普通的例子里,你可以把engine看作類的內(nèi)部隱式方法.
ruby>classTest
|deftimes_two(a)
|printa,"timestwois",engine(a),"\n"
|end
|defengine(b)
|b*2
|end
|private:engine#thishidesenginefromusers
|end
Test
ruby>test=Test.new
#
ruby>test.engine(6)
ERR:(eval):1:privatemethod`engine'calledfor#
ruby>test.times_two(6)
6timestwois12.
nil
開始時(shí),我們本希望test.engine(6)返回12,但隨后當(dāng)我們扮演一個(gè)Test對(duì)象用戶時(shí)我們了解到engine是不可訪問的(inaccessible).只有Test的其它方法,像times_two可以使用engine.我們被要求保持程序的外部接口,也就是times_two這些方法.管理此類的程序員可以自由的更改engine(這里,可能把b*2改為b+b,假設(shè)這樣可以提高性能)而不影響用戶與Test對(duì)象打交道.這個(gè)例子當(dāng)然過于簡(jiǎn)單;訪問控制的優(yōu)點(diǎn)只有當(dāng)我們開始寫更復(fù)雜和有趣的類時(shí)方能顯現(xiàn)出來.