欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解Ruby中的單件方法和單件類(lèi)

 更新時(shí)間:2016年06月04日 15:53:17   作者:日拱一卒  
單件方法顧名思義,就是只作用于單個(gè)對(duì)象的方法,同理單件類(lèi)就是單件方法所存在的類(lèi),規(guī)定其作用域,這里我們就來(lái)詳解Ruby中的單件方法和單件類(lèi):

單件方法

Ruby允許給單個(gè)對(duì)象增加方法,這種只針對(duì)單個(gè)對(duì)象生效的方法,稱(chēng)為單件方法

示例代碼

str = “just a regular string”

def str.title?
  self.upcase == self
end

str.title? # => false
str.methods.grep(/title?/) # => [:title?]
str.singleton_methods  #=> [:title?]

str.class # => String
String.title? #=> NoMethodError

另外,除了上面使用的定義方法,還可以通過(guò)Object#define_singleton_method方法來(lái)定義單件方法

單件方法與類(lèi)方法

前面的筆記中有說(shuō)道在Ruby中類(lèi)也是對(duì)象,而類(lèi)名只是常量,所以,在類(lèi)上調(diào)用方法其實(shí)跟在對(duì)象上調(diào)用方法一樣:

類(lèi)方法的實(shí)質(zhì)是: 它是一個(gè)類(lèi)的單件方法,實(shí)際上,如果比較單件方法的定義和類(lèi)方法的定義,會(huì)發(fā)現(xiàn)其實(shí)二者是一樣的
 

def obj.a_singleton_method; end
def MyClass.another_class_method; end

二者均使用了def關(guān)鍵詞做定義

def object.method
  #方法主體
end

上面的object可以是*對(duì)象的引用、常量類(lèi)名或者self。

類(lèi)宏attr_accessor

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

示例代碼

class MyClass
  def my_attribute=(value)
    @my_attribute =value  
  end
  def my_attribute
    @my_attribute
  end
end

obj = MyClass.new
obj.my_attribute = ‘x'
obj.my_attribute  #=> ‘x'

但是上面這種寫(xiě)法,如果屬性眾多的話就會(huì)存在Repeat Yourself的地方,這時(shí)就可以用到下面三個(gè)類(lèi)宏:

  • Module#attr_reader 生成一個(gè)讀方法
  • Module#attr_writer 生成一個(gè)寫(xiě)方法
  • Module#attr_accessor 同時(shí)生成讀方法和寫(xiě)方法

示例代碼

class MyClass
  attr_accessor :my_attribue
end

這樣是不是就簡(jiǎn)潔多了呢? 當(dāng)然,使用方法(讀與寫(xiě))跟上面的實(shí)現(xiàn)是一致的。

單件類(lèi)

我們知道Ruby中對(duì)象的方法的查找順序是: 先向右,再向上,其含義就是先向右找到對(duì)象的類(lèi),先在類(lèi)的實(shí)例方法中嘗試查找,如果沒(méi)有找到,再繼續(xù)順著祖先鏈找。

前面一篇中有介紹過(guò)單件方法,單件方法是指那些只針對(duì)某個(gè)對(duì)象有效的方法,那么如果為一個(gè)對(duì)象定義了單件方法,那么這個(gè)單件方法的查找順序又應(yīng)該是怎樣的?

class MyClass
  def my_method; end
end

obj = MyClass.new

def obj.my_singleton_method; end

首先,單件方法不會(huì)在obj中,因?yàn)閛bj不是一個(gè)類(lèi),其次它也不在MyClass中,那樣的話所有的MyClass都應(yīng)該能共享調(diào)用這個(gè)方法,也就構(gòu)不成單件類(lèi)了。同理,單件方法也不能在祖先鏈的某個(gè)位置(類(lèi)似superclass: Object)中。正確的位置是在單件類(lèi)中,這個(gè)類(lèi)其實(shí)就是我們?cè)趇rb中向?qū)ο笤?xún)問(wèn)它的類(lèi)時(shí)(obj.class)得到的那個(gè)類(lèi),不同的是這類(lèi)與普通的類(lèi)還是有稍稍不同的。也可以稱(chēng)其為元類(lèi)或本征類(lèi)。

打開(kāi)單件類(lèi)

Ruby提供了兩種方法獲取單件類(lèi)的引用,一種是通過(guò)傳統(tǒng)的關(guān)鍵詞class配合特殊的語(yǔ)法

法一
 

class << an_object
  # 自己的代碼
end

obj = Object.new
singleton_class = class << obj
  self
end
singleton_class.class # => Class

另一個(gè)方法是,通過(guò)Object#singleton_class方法來(lái)獲得單件類(lèi)的引用:

法二

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

 
單件類(lèi)的特性

  • 每個(gè)單件類(lèi)只有一個(gè)實(shí)例(被稱(chēng)為單件類(lèi)的原因),而且不能被繼承
  • 單件類(lèi)是一個(gè)對(duì)象的單件方法的存活所在
  • 引入單件類(lèi)后的方法查找

基于上面對(duì)單件類(lèi)的基本認(rèn)識(shí),引入單件類(lèi)后,Ruby的方法查找方式就不應(yīng)該是先從其類(lèi)(普通類(lèi))開(kāi)始,而是應(yīng)該先從對(duì)象的單件類(lèi)中開(kāi)始查找,如果在單件類(lèi)中沒(méi)有找到想要的方法,它才會(huì)開(kāi)始沿著類(lèi)(普通類(lèi))開(kāi)始,再到祖先鏈上去找。這樣從單件類(lèi)之后開(kāi)始,一切又回到了我們?cè)跊](méi)有引入單件類(lèi)時(shí)候的次序。

通過(guò)下面這個(gè)代碼可以自行驗(yàn)證一下

class C
  def a_method
    ‘C#a_method()'
  end
end

class D < C; end

obj = D.new

打開(kāi)單件類(lèi)定義單件方法

class << obj
  def a_singleton_method
    ‘obj#a_singleton_method()'
  end
end

obj.singleton_class.superclass #=> D

相關(guān)文章

最新評(píng)論