詳解Ruby中的單件方法和單件類_ruby專題

來源:互聯網
上載者:User

單件方法

Ruby允許給單個對象增加方法,這種只針對單個對象生效的方法,稱為單件方法

範例程式碼

str = “just a regular string”def str.title?  self.upcase == selfendstr.title? # => falsestr.methods.grep(/title?/) # => [:title?]str.singleton_methods  #=> [:title?]str.class # => StringString.title? #=> NoMethodError

另外,除了上面使用的定義方法,還可以通過Object#define_singleton_method方法來定義單件方法

單件方法與類方法

前面的筆記中有說道在Ruby中類也是對象,而類名只是常量,所以,在類上調用方法其實跟在對象上調用方法一樣:

類方法的實質是: 它是一個類的單件方法,實際上,如果比較單件方法的定義和類方法的定義,會發現其實二者是一樣的
 

def obj.a_singleton_method; enddef MyClass.another_class_method; end

二者均使用了def關鍵詞做定義

def object.method  #方法主體end

上面的object可以是*對象的引用、常量類名或者self。

類宏attr_accessor

Ruby對象沒有屬性,如果希望得到一些像屬性的東西,需要分別定義一個讀方法和寫方法(也就是java、objc中的set和get方法),最直接的可以這樣:

範例程式碼

class MyClass  def my_attribute=(value)    @my_attribute =value    end  def my_attribute    @my_attribute  endendobj = MyClass.newobj.my_attribute = ‘x'obj.my_attribute  #=> ‘x'

但是上面這種寫法,如果屬性眾多的話就會存在Repeat Yourself的地方,這時就可以用到下面三個類宏:

  • Module#attr_reader 產生一個讀方法
  • Module#attr_writer 產生一個寫方法
  • Module#attr_accessor 同時產生讀方法和寫方法

範例程式碼

class MyClass  attr_accessor :my_attribueend

這樣是不是就簡潔多了呢? 當然,使用方法(讀與寫)跟上面的實現是一致的。

單件類

我們知道Ruby中對象的方法的尋找順序是: 先向右,再向上,其含義就是先向右找到對象的類,先在類的執行個體方法中嘗試尋找,如果沒有找到,再繼續順著祖先鏈找。

前面一篇中有介紹過單件方法,單件方法是指那些只針對某個對象有效方法,那麼如果為一個對象定義了單件方法,那麼這個單件方法的尋找順序又應該是怎樣的?

class MyClass  def my_method; endendobj = MyClass.newdef obj.my_singleton_method; end

首先,單件方法不會在obj中,因為obj不是一個類,其次它也不在MyClass中,那樣的話所有的MyClass都應該能共用調用這個方法,也就構不成單件類了。同理,單件方法也不能在祖先鏈的某個位置(類似superclass: Object)中。正確的位置是在單件類中,這個類其實就是我們在irb中向對象詢問它的類時(obj.class)得到的那個類,不同的是這類與普通的類還是有稍稍不同的。也可以稱其為元類或本徵類。

開啟單件類

Ruby提供了兩種方法擷取單件類的引用,一種是通過傳統的關鍵詞class配合特殊的文法

法一
 

class << an_object  # 自己的代碼endobj = Object.newsingleton_class = class << obj  selfendsingleton_class.class # => Class

另一個方法是,通過Object#singleton_class方法來獲得單件類的引用:

法二

“abc”.singleton_class  # => #<Class: #<String:0xxxxxx>>

 
單件類的特性

  • 每個單件類只有一個執行個體(被稱為單件類的原因),而且不能被繼承
  • 單件類是一個對象的單件方法的存活所在
  • 引入單件類後的方法尋找

基於上面對單件類的基本認識,引入單件類後,Ruby的方法尋找方式就不應該是先從其類(普通類)開始,而是應該先從對象的單件類中開始尋找,如果在單件類中沒有找到想要的方法,它才會開始沿著類(普通類)開始,再到祖先鏈上去找。這樣從單件類之後開始,一切又回到了我們在沒有引入單件類時候的次序。

通過下面這個代碼可以自行驗證一下

class C  def a_method    ‘C#a_method()'  endendclass D < C; endobj = D.new

開啟單件類定義單件方法

class << obj  def a_singleton_method    ‘obj#a_singleton_method()'  endendobj.singleton_class.superclass #=> D

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.