打開類
可以重新打開已經(jīng)存在的類并對之進(jìn)行動態(tài)修改,即使像String或者Array這樣標(biāo)準(zhǔn)庫的類也不例外。這種行為方式稱之為打開類(open class)
猴子補(bǔ)丁
如果你粗心地為某個類添加了新功能,同時覆蓋了類原來的功能,進(jìn)而影響到其他部分的代碼,這樣的patch稱之為猴子補(bǔ)丁(Monkeypatch)
類與模塊
Ruby的class關(guān)鍵字更像是一個作用域操作符,而不是類型聲明語句。class關(guān)鍵字的核心任務(wù)是把你帶到類的上下文中,讓你可以在里面定義方法。
每個類都是一個模塊,類就是帶有三個方法(new,allocate,superclass)的增強(qiáng)模塊,通過這三個方法可以組織類的繼承結(jié)構(gòu),并創(chuàng)建對象
Ruby中的類和模塊的概念十分接近,完全可以將二者相互替代,之所以同時保留二者的原因是為了保持代碼的清晰性,讓代碼意圖更加明確。使用原則:
Ruby中的::符號
Ruby中常量的路徑(作用域),類似與文件系統(tǒng)中的目錄,通過::進(jìn)行分割和訪問,默認(rèn)直接以::開頭(例: :: Y)表示變量路徑的根位置
什么是對象
對象就是一組實(shí)例變量外加一個指向其類的引用。對象的方法并不存在于對象本身,而是存在于對象的類中。
什么是類
類就是一個對象(Class類的一個實(shí)例)外加一組實(shí)例方法和一個對其超類的引用。Class類是Module類的子類,因此一個類也是一個模塊。
load與require方法的異同
通過load和require都可以進(jìn)行導(dǎo)入別人的代碼,不同的是load方法用來加載代碼,如果不希望污染當(dāng)前的命名空間,需要通過load(‘file.rb',true)顯式的要求創(chuàng)建一個匿名模塊來,接管file.rb的常量,require用于導(dǎo)入類庫,此外,就加載次數(shù)上load方法每次調(diào)用都會再次運(yùn)行所加載文件,require則對每個庫文件只加載一次。
prepend、include與祖先鏈
祖先鏈用于描述Ruby對象的繼承關(guān)系,因?yàn)轭惻c模塊是父子關(guān)系,所以祖先鏈中也可以包含模塊,prepend與include分別可以向鏈中添加模塊,不同的是調(diào)用include方法,模塊會被插入祖先鏈,當(dāng)前類的正上方,而prepend同樣是插入到祖先鏈,但位置其他卻在當(dāng)前類的正下方,另外通過Class.ancestors可以查看當(dāng)前的祖先鏈
private規(guī)則
不能通過明確指定接受者來調(diào)用私有方法。私有方法只能通過隱性的接受者self調(diào)用(Object#send是個例外)
self相關(guān)
調(diào)用一個方法時,接受者會扮演self角色 任何沒有明確指定接受者的方法調(diào)用,都當(dāng)做是調(diào)用self的方法 定義一個模塊(或類)時,該模塊(或類)會扮演self角色
對象、類與模塊之間關(guān)系
上面Module.class指向的也是Class類,可以理解為上面方框內(nèi)容均為Class,但他們的父子組織關(guān)系通過superclass建立并存在異同,可以通過Class.ancestors查看。
動態(tài)方法
動態(tài)調(diào)用方法
在Ruby中通過Object#send方法可以代替點(diǎn)標(biāo)識調(diào)用對象的指定實(shí)例方法
示例代碼
class MyClass def my_method(my_arg) my_arg * 2 end end obj = MyClass.new obj.my_method(3) #=> 6 obj.send(:my_method, 3) #=> 6
上面代碼通過直接調(diào)用和使用send方法調(diào)用得到的結(jié)果是一樣的,使用send的好處是,可以在編碼中,動態(tài)的決定方法調(diào)用。這個技巧在元編程中被稱為動態(tài)派發(fā)
另外需要指出的地方是通過Object#send不僅可以調(diào)用公共方法,也可以調(diào)用對象的私有方法。如果想保留對象的封裝特性,不向外暴露私有方法可以使用Object#public_send方法。
動態(tài)定義方法
除了方法的動態(tài)調(diào)用之外,Ruby還通過Module#define_method方法和代碼塊提供了動態(tài)方法定義方式
示例代碼
class MyClass define_method :my_method do |my_arg| my_arg * 3 do end obj = MyClass.new obj.my_method(2) #=> 6
上面代碼通過define_method方法取代了關(guān)鍵詞def,其本質(zhì)上都是相同的,只是在定義方式上,define_method的方式更加靈活一些,可以通過在編碼中通過推導(dǎo),完成函數(shù)的定義,增加了實(shí)現(xiàn)的靈活性。
method_missing方法
嚴(yán)格意義上將method_missing方法,并不算是明確的定義(不會出現(xiàn)在methods列表中),其本質(zhì)是通過方法查找的機(jī)制來截獲調(diào)用信息進(jìn)而合理的給出相應(yīng)方法的回應(yīng)。有點(diǎn)類似與異常處理中的拋出異常,一層一層的往外拋。
method_missing利用的機(jī)制是,當(dāng)一個對象進(jìn)行某個方法調(diào)用的時候,會到其對應(yīng)的類的實(shí)例方法中進(jìn)行查找,如果沒有找到,則順著祖先鏈向上查找,直到找到BasicObject類為止。如果都沒有則會最終調(diào)用一個BasicObject#method_missing拋出NoMethodError異常。
當(dāng)我們需要定義很多相似的方法時候,可以通過重寫method_missing方法,對相似的方法進(jìn)行統(tǒng)一做出回應(yīng),這樣一來其行為就類似與調(diào)用定義過的方法一樣。
示例代碼
class Roulette def method_missing(name, *args) person = name.to_s.capitalize super unless %w[Bob Frank Bill Honda Eric].include? person number = 0 3.times do number = rand(10) + 1 puts "#{number}..." end "#{person} got a #{number}" end end number_of = Roulette.new puts number_of.bob puts number_of.kitty
動態(tài)代理
對一些封裝過的對象,通過method_missing方法收集調(diào)用,并把這些調(diào)用轉(zhuǎn)發(fā)到被封裝的對象,這一過程稱為動態(tài)代理,其中method_missing體現(xiàn)了動態(tài),轉(zhuǎn)發(fā)體現(xiàn)了代理
const_missing方法
與method_missing類似,還有關(guān)于常量的const_missing方法,當(dāng)引用一個不存在的常量時,Ruby會把這個常量名作為一個符號傳遞給const_missing方法。
白板類(blank slates)
擁有極少方法的類稱為白板類,通過繼承BasicObject類,可以迅速的得到一個白板類。除了這種方法以外,還可以通過刪除方法來將一個普通類變?yōu)榘装孱悺?/p>
刪除方法
刪除某個方法有兩種方式:
二者的區(qū)別是Module#undef_method會刪除所有(包括繼承而來的)方法。而Module#remove_method只刪除接受者自己的方法,而保留繼承來的方法。
動態(tài)方法與Method_missing的使用原則
當(dāng)可以使用動態(tài)方法時候,盡量使用動態(tài)方法。除非必須使用method_missing方法(方法特別多的情況),否則盡量少使用它。
標(biāo)簽:自貢 鄂州 六安 松原 淄博 河北 宜春 石嘴山
巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Ruby的面向?qū)ο蠓绞骄幊虒W(xué)習(xí)雜記》,本文關(guān)鍵詞 Ruby,的,面向,對象,方式,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。