標籤:style blog color 使用 代碼 div c++ new
上一篇講了ruby中的可調用對象proc和lambda,他們都是塊轉換成的對象。ruby中的可調用對象還有方法。通過使用method方法,並且以方法名作為參數(字串或者符號),就可以得到一個方法對象。
如下:
1 class C2 def talk3 p "hello world"4 end5 end6 7 c = C.new8 meth = c.method :talk9 meth.call
輸出: "hello world"
類C有個執行個體方法talk。類C的執行個體c調用方法method,並且用:talk做參數,得到一個方法對象meth。然後meth調用call,輸出"hello world"。
值得注意的是,meth這個方法對象是綁定在對象c上的方法。調用call的時候,也知道是c在調用talk方法。
可以用Method#unbind()方法來把一個方法跟它綁定的對象相分離,該方法返回一個UnboundMethod對象,不能執行UnboundMethod對象,必須把它綁定到一個對象上,使之再次成為一個Method對象。
如下:(在上面代碼的基礎上)
1 class D < C2 end3 d = D.new4 unbound = meth.unbind5 new_meth = unbound.bind(d)6 new_meth.call
輸出: "hello world"
unbind解除綁定之後,綁定到C的子類D的一個執行個體對象d上。再對方法對象調用call。如果不想對一個已綁定的方法進行解除綁定,可以用如下的代碼得到解除綁定對象:
unbound = C.instance_method (:talk)
為什麼要這樣呢?存在總是合理的,綁定與解除綁定也有它的適用之處。
在之前《ruby中的方法尋找》中,我們知道ruby的方法調用是根據匹配到的第一個方法來調用的。那麼如果要調用匹配路徑上的匹配到的第二個方法呢?ruby中有super這個關鍵字。但是加入super關鍵字相當於改變了這個方法。它並不是直接調用了匹配路徑上的第二個方法。而是在匹配到的第一個方法裡調用第二個方法。還有,如果要調用匹配路徑上的第三個匹配到的方法呢?第四個呢?這個就可以用到綁定的方法了。
如下:
1 class C 2 def method 3 p "this is in C" 4 end 5 end 6 7 class D < C 8 def method 9 p "this is in D"10 end11 end12 13 class E < D14 def method 15 p "this is in E"16 end17 end18 19 e = E.new20 e.method21 22 D.instance_method(:method).bind(e).call23 C.instance_method(:method).bind(e).call
輸出:
"this is in E"
"this is in D"
"this is in C"
如代碼所示,定義類C,D,E,D繼承C,E繼承D。他們都定義了方法method。那麼E的執行個體e的方法尋找路徑上這三個method都可以調用。如果直接調用e.method。那麼肯定是調用了第一個匹配的方法,也就是類E中的方法,如輸出的第一行。如果要調用類D中的方法method呢?如代碼22行。把D的執行個體方法綁定到對象e上,然後在調用。調用類C中的方法method,同理。如23行。
學過C++的話,看到這個可能會想到衍生類別指標引用基類對象。衍生類別是基類的特例,所以只有強制轉換類型之後才能引用基類的對象。和ruby中的綁定類似,只有和父類的方法綁定以後,才能調用父類的方法。
和上一篇一起做一個總結。ruby中可以被調用的對象有:
塊(不算是對象,可以被調用),proc(Proc類的對象),lambda(Proc類的對象),這三種都是在定義它們的範圍中執行。
方法:綁定於對象,在所綁定的對象的範圍中執行。
將一種對象轉換成另一種對象的方法有:Proc.new(),Method#to_proc(),&操作符