打開類
可以重新打開已經存在的類并對之進行動態(tài)修改,即使像String或者Array這樣標準庫的類也不例外。這種行為方式稱之為打開類(open class)
猴子補丁
如果你粗心地為某個類添加了新功能,同時覆蓋了類原來的功能,進而影響到其他部分的代碼,這樣的patch稱之為猴子補丁(Monkeypatch)
類與模塊
Ruby的class關鍵字更像是一個作用域操作符,而不是類型聲明語句。class關鍵字的核心任務是把你帶到類的上下文中,讓你可以在里面定義方法。
每個類都是一個模塊,類就是帶有三個方法(new,allocate,superclass)的增強模塊,通過這三個方法可以組織類的繼承結構,并創(chuàng)建對象
Ruby中的類和模塊的概念十分接近,完全可以將二者相互替代,之所以同時保留二者的原因是為了保持代碼的清晰性,讓代碼意圖更加明確。使用原則:
- 希望把自己代碼包含(include)到別的代碼中,應該使用模塊
- 希望某段代碼被實例化或被繼承,應該使用類
- 模塊機制可以用來實現類似其它語言中的命名空間(Namespace)概念
Ruby中的::符號
Ruby中常量的路徑(作用域),類似與文件系統(tǒng)中的目錄,通過::進行分割和訪問,默認直接以::開頭(例: :: Y)表示變量路徑的根位置
什么是對象
對象就是一組實例變量外加一個指向其類的引用。對象的方法并不存在于對象本身,而是存在于對象的類中。
什么是類
類就是一個對象(Class類的一個實例)外加一組實例方法和一個對其超類的引用。Class類是Module類的子類,因此一個類也是一個模塊。
load與require方法的異同
通過load和require都可以進行導入別人的代碼,不同的是load方法用來加載代碼,如果不希望污染當前的命名空間,需要通過load(‘file.rb',true)顯式的要求創(chuàng)建一個匿名模塊來,接管file.rb的常量,require用于導入類庫,此外,就加載次數上load方法每次調用都會再次運行所加載文件,require則對每個庫文件只加載一次。
prepend、include與祖先鏈
祖先鏈用于描述Ruby對象的繼承關系,因為類與模塊是父子關系,所以祖先鏈中也可以包含模塊,prepend與include分別可以向鏈中添加模塊,不同的是調用include方法,模塊會被插入祖先鏈,當前類的正上方,而prepend同樣是插入到祖先鏈,但位置其他卻在當前類的正下方,另外通過Class.ancestors可以查看當前的祖先鏈
private規(guī)則
不能通過明確指定接受者來調用私有方法。私有方法只能通過隱性的接受者self調用(Object#send是個例外)
self相關
調用一個方法時,接受者會扮演self角色 任何沒有明確指定接受者的方法調用,都當做是調用self的方法 定義一個模塊(或類)時,該模塊(或類)會扮演self角色
對象、類與模塊之間關系

上面Module.class指向的也是Class類,可以理解為上面方框內容均為Class,但他們的父子組織關系通過superclass建立并存在異同,可以通過Class.ancestors查看。
動態(tài)方法
動態(tài)調用方法
在Ruby中通過Object#send方法可以代替點標識調用對象的指定實例方法
示例代碼
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
上面代碼通過直接調用和使用send方法調用得到的結果是一樣的,使用send的好處是,可以在編碼中,動態(tài)的決定方法調用。這個技巧在元編程中被稱為動態(tài)派發(fā)
另外需要指出的地方是通過Object#send不僅可以調用公共方法,也可以調用對象的私有方法。如果想保留對象的封裝特性,不向外暴露私有方法可以使用Object#public_send方法。
動態(tài)定義方法
除了方法的動態(tài)調用之外,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方法取代了關鍵詞def,其本質上都是相同的,只是在定義方式上,define_method的方式更加靈活一些,可以通過在編碼中通過推導,完成函數的定義,增加了實現的靈活性。
method_missing方法
嚴格意義上將method_missing方法,并不算是明確的定義(不會出現在methods列表中),其本質是通過方法查找的機制來截獲調用信息進而合理的給出相應方法的回應。有點類似與異常處理中的拋出異常,一層一層的往外拋。
method_missing利用的機制是,當一個對象進行某個方法調用的時候,會到其對應的類的實例方法中進行查找,如果沒有找到,則順著祖先鏈向上查找,直到找到BasicObject類為止。如果都沒有則會最終調用一個BasicObject#method_missing拋出NoMethodError異常。
當我們需要定義很多相似的方法時候,可以通過重寫method_missing方法,對相似的方法進行統(tǒng)一做出回應,這樣一來其行為就類似與調用定義過的方法一樣。
示例代碼
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方法收集調用,并把這些調用轉發(fā)到被封裝的對象,這一過程稱為動態(tài)代理,其中method_missing體現了動態(tài),轉發(fā)體現了代理
const_missing方法
與method_missing類似,還有關于常量的const_missing方法,當引用一個不存在的常量時,Ruby會把這個常量名作為一個符號傳遞給const_missing方法。
白板類(blank slates)
擁有極少方法的類稱為白板類,通過繼承BasicObject類,可以迅速的得到一個白板類。除了這種方法以外,還可以通過刪除方法來將一個普通類變?yōu)榘装孱悺?/p>
刪除方法
刪除某個方法有兩種方式:
- Module#undef_method
- Module#remove_method
二者的區(qū)別是Module#undef_method會刪除所有(包括繼承而來的)方法。而Module#remove_method只刪除接受者自己的方法,而保留繼承來的方法。
動態(tài)方法與Method_missing的使用原則
當可以使用動態(tài)方法時候,盡量使用動態(tài)方法。除非必須使用method_missing方法(方法特別多的情況),否則盡量少使用它。
您可能感興趣的文章:- Ruby面向對象編程中類的方法與類的擴展
- Ruby面向對象編程中類與方法的基礎學習
- 簡要解讀Ruby面向對象編程中的作用域
- Ruby的面向對象編程的基礎教程
- Ruby面向對象編程詳解
- ruby 面向對象思維 概念
- Ruby 面向對象知識總結