主頁(yè) > 知識(shí)庫(kù) > Ruby的面向?qū)ο蠓绞骄幊虒W(xué)習(xí)雜記

Ruby的面向?qū)ο蠓绞骄幊虒W(xué)習(xí)雜記

熱門(mén)標(biāo)簽:電話機(jī)器人電話卡封號(hào)怎么辦 樂(lè)昌電話機(jī)器人 真人語(yǔ)音電銷機(jī)器人系統(tǒng) 北京語(yǔ)音電銷機(jī)器人價(jià)格 買(mǎi)了外呼系統(tǒng)不想用了怎么辦 開(kāi)封百應(yīng)電銷機(jī)器人聯(lián)系方式 邯鄲外呼調(diào)研線路 浦東上海400開(kāi)頭的電話申請(qǐng) 武漢呼叫中心外呼系統(tǒng)線路商

打開(kāi)類

可以重新打開(kāi)已經(jīng)存在的類并對(duì)之進(jìn)行動(dòng)態(tài)修改,即使像String或者Array這樣標(biāo)準(zhǔn)庫(kù)的類也不例外。這種行為方式稱之為打開(kāi)類(open class)

猴子補(bǔ)丁

如果你粗心地為某個(gè)類添加了新功能,同時(shí)覆蓋了類原來(lái)的功能,進(jìn)而影響到其他部分的代碼,這樣的patch稱之為猴子補(bǔ)丁(Monkeypatch)

類與模塊

Ruby的class關(guān)鍵字更像是一個(gè)作用域操作符,而不是類型聲明語(yǔ)句。class關(guān)鍵字的核心任務(wù)是把你帶到類的上下文中,讓你可以在里面定義方法。

每個(gè)類都是一個(gè)模塊,類就是帶有三個(gè)方法(new,allocate,superclass)的增強(qiáng)模塊,通過(guò)這三個(gè)方法可以組織類的繼承結(jié)構(gòu),并創(chuàng)建對(duì)象

Ruby中的類和模塊的概念十分接近,完全可以將二者相互替代,之所以同時(shí)保留二者的原因是為了保持代碼的清晰性,讓代碼意圖更加明確。使用原則:

  • 希望把自己代碼包含(include)到別的代碼中,應(yīng)該使用模塊
  • 希望某段代碼被實(shí)例化或被繼承,應(yīng)該使用類
  • 模塊機(jī)制可以用來(lái)實(shí)現(xiàn)類似其它語(yǔ)言中的命名空間(Namespace)概念

Ruby中的::符號(hào)

Ruby中常量的路徑(作用域),類似與文件系統(tǒng)中的目錄,通過(guò)::進(jìn)行分割和訪問(wèn),默認(rèn)直接以::開(kāi)頭(例: :: Y)表示變量路徑的根位置

什么是對(duì)象

對(duì)象就是一組實(shí)例變量外加一個(gè)指向其類的引用。對(duì)象的方法并不存在于對(duì)象本身,而是存在于對(duì)象的類中。

什么是類

類就是一個(gè)對(duì)象(Class類的一個(gè)實(shí)例)外加一組實(shí)例方法和一個(gè)對(duì)其超類的引用。Class類是Module類的子類,因此一個(gè)類也是一個(gè)模塊。

load與require方法的異同

通過(guò)load和require都可以進(jìn)行導(dǎo)入別人的代碼,不同的是load方法用來(lái)加載代碼,如果不希望污染當(dāng)前的命名空間,需要通過(guò)load(‘file.rb',true)顯式的要求創(chuàng)建一個(gè)匿名模塊來(lái),接管file.rb的常量,require用于導(dǎo)入類庫(kù),此外,就加載次數(shù)上load方法每次調(diào)用都會(huì)再次運(yùn)行所加載文件,require則對(duì)每個(gè)庫(kù)文件只加載一次。

prepend、include與祖先鏈

祖先鏈用于描述Ruby對(duì)象的繼承關(guān)系,因?yàn)轭惻c模塊是父子關(guān)系,所以祖先鏈中也可以包含模塊,prepend與include分別可以向鏈中添加模塊,不同的是調(diào)用include方法,模塊會(huì)被插入祖先鏈,當(dāng)前類的正上方,而prepend同樣是插入到祖先鏈,但位置其他卻在當(dāng)前類的正下方,另外通過(guò)Class.ancestors可以查看當(dāng)前的祖先鏈

private規(guī)則

不能通過(guò)明確指定接受者來(lái)調(diào)用私有方法。私有方法只能通過(guò)隱性的接受者self調(diào)用(Object#send是個(gè)例外)

self相關(guān)

調(diào)用一個(gè)方法時(shí),接受者會(huì)扮演self角色 任何沒(méi)有明確指定接受者的方法調(diào)用,都當(dāng)做是調(diào)用self的方法 定義一個(gè)模塊(或類)時(shí),該模塊(或類)會(huì)扮演self角色

對(duì)象、類與模塊之間關(guān)系

上面Module.class指向的也是Class類,可以理解為上面方框內(nèi)容均為Class,但他們的父子組織關(guān)系通過(guò)superclass建立并存在異同,可以通過(guò)Class.ancestors查看。

動(dòng)態(tài)方法

動(dòng)態(tài)調(diào)用方法

在Ruby中通過(guò)Object#send方法可以代替點(diǎn)標(biāo)識(shí)調(diào)用對(duì)象的指定實(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

上面代碼通過(guò)直接調(diào)用和使用send方法調(diào)用得到的結(jié)果是一樣的,使用send的好處是,可以在編碼中,動(dòng)態(tài)的決定方法調(diào)用。這個(gè)技巧在元編程中被稱為動(dòng)態(tài)派發(fā)

另外需要指出的地方是通過(guò)Object#send不僅可以調(diào)用公共方法,也可以調(diào)用對(duì)象的私有方法。如果想保留對(duì)象的封裝特性,不向外暴露私有方法可以使用Object#public_send方法。

動(dòng)態(tài)定義方法

除了方法的動(dòng)態(tài)調(diào)用之外,Ruby還通過(guò)Module#define_method方法和代碼塊提供了動(dòng)態(tài)方法定義方式

示例代碼

class MyClass
  define_method :my_method do |my_arg|
    my_arg * 3
  do
end

obj = MyClass.new
obj.my_method(2) #=> 6

上面代碼通過(guò)define_method方法取代了關(guān)鍵詞def,其本質(zhì)上都是相同的,只是在定義方式上,define_method的方式更加靈活一些,可以通過(guò)在編碼中通過(guò)推導(dǎo),完成函數(shù)的定義,增加了實(shí)現(xiàn)的靈活性。

method_missing方法

嚴(yán)格意義上將method_missing方法,并不算是明確的定義(不會(huì)出現(xiàn)在methods列表中),其本質(zhì)是通過(guò)方法查找的機(jī)制來(lái)截獲調(diào)用信息進(jìn)而合理的給出相應(yīng)方法的回應(yīng)。有點(diǎn)類似與異常處理中的拋出異常,一層一層的往外拋。

method_missing利用的機(jī)制是,當(dāng)一個(gè)對(duì)象進(jìn)行某個(gè)方法調(diào)用的時(shí)候,會(huì)到其對(duì)應(yīng)的類的實(shí)例方法中進(jìn)行查找,如果沒(méi)有找到,則順著祖先鏈向上查找,直到找到BasicObject類為止。如果都沒(méi)有則會(huì)最終調(diào)用一個(gè)BasicObject#method_missing拋出NoMethodError異常。

當(dāng)我們需要定義很多相似的方法時(shí)候,可以通過(guò)重寫(xiě)method_missing方法,對(duì)相似的方法進(jìn)行統(tǒng)一做出回應(yīng),這樣一來(lái)其行為就類似與調(diào)用定義過(guò)的方法一樣。

示例代碼

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

動(dòng)態(tài)代理

對(duì)一些封裝過(guò)的對(duì)象,通過(guò)method_missing方法收集調(diào)用,并把這些調(diào)用轉(zhuǎn)發(fā)到被封裝的對(duì)象,這一過(guò)程稱為動(dòng)態(tài)代理,其中method_missing體現(xiàn)了動(dòng)態(tài),轉(zhuǎn)發(fā)體現(xiàn)了代理

const_missing方法

與method_missing類似,還有關(guān)于常量的const_missing方法,當(dāng)引用一個(gè)不存在的常量時(shí),Ruby會(huì)把這個(gè)常量名作為一個(gè)符號(hào)傳遞給const_missing方法。

白板類(blank slates)

擁有極少方法的類稱為白板類,通過(guò)繼承BasicObject類,可以迅速的得到一個(gè)白板類。除了這種方法以外,還可以通過(guò)刪除方法來(lái)將一個(gè)普通類變?yōu)榘装孱悺?/p>

刪除方法

刪除某個(gè)方法有兩種方式:

  • Module#undef_method
  • Module#remove_method

二者的區(qū)別是Module#undef_method會(huì)刪除所有(包括繼承而來(lái)的)方法。而Module#remove_method只刪除接受者自己的方法,而保留繼承來(lái)的方法。

動(dòng)態(tài)方法與Method_missing的使用原則

當(dāng)可以使用動(dòng)態(tài)方法時(shí)候,盡量使用動(dòng)態(tài)方法。除非必須使用method_missing方法(方法特別多的情況),否則盡量少使用它。

您可能感興趣的文章:
  • Ruby面向?qū)ο缶幊讨蓄惖姆椒ㄅc類的擴(kuò)展
  • Ruby面向?qū)ο缶幊讨蓄惻c方法的基礎(chǔ)學(xué)習(xí)
  • 簡(jiǎn)要解讀Ruby面向?qū)ο缶幊讨械淖饔糜?/li>
  • Ruby的面向?qū)ο缶幊痰幕A(chǔ)教程
  • Ruby面向?qū)ο缶幊淘斀?/li>
  • ruby 面向?qū)ο笏季S 概念
  • Ruby 面向?qū)ο笾R(shí)總結(jié)

標(biāo)簽:自貢 鄂州 六安 松原 淄博 河北 宜春 石嘴山

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Ruby的面向?qū)ο蠓绞骄幊虒W(xué)習(xí)雜記》,本文關(guān)鍵詞  Ruby,的,面向,對(duì)象,方式,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《Ruby的面向?qū)ο蠓绞骄幊虒W(xué)習(xí)雜記》相關(guān)的同類信息!
  • 本頁(yè)收集關(guān)于Ruby的面向?qū)ο蠓绞骄幊虒W(xué)習(xí)雜記的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章