佳木斯湛栽影视文化发展公司

主頁 > 知識庫 > 詳解Ruby中的instance_eval方法及其與class_eval的對比

詳解Ruby中的instance_eval方法及其與class_eval的對比

熱門標(biāo)簽:百度AI接口 電話運營中心 客戶服務(wù) 語音系統(tǒng) Win7旗艦版 呼叫中心市場需求 硅谷的囚徒呼叫中心 企業(yè)做大做強

instance_eval方法

這個BasicObject#instance_eval有點類似JS中的bind方法,不同的時,bind是將this傳入到對象中,而instance_eval則是將代碼塊(上下文探針Context Probe)傳入到指定的對象中,一個是傳對象,一個是傳執(zhí)行體。通過這種方式就可以在instance_eval中的代碼塊里訪問到調(diào)用者對象中的變量。

示例代碼

class MyClass
  def initialize
    @v = 1
  end
end
obj = MyClass.new

obj.instance_eval do
  self  #=> #MyClass:0x33333 @v=1>
  @v   #=> 1 
end

v = 2
obj.instance_eval { @v = v }
obj.instance_eval { @v }  # => 2

此外,instance_eval方法還有一個雙胞胎兄弟:instance_exec方法。相比前者后者更加靈活,允許對代碼塊傳入?yún)?shù)。

示例代碼

class C
  def initialize
    @x = 1
  end
end
class D
  def twisted_method
    @y = 2
    #C.new.instance_eval { “@x: #{@x}, @y>: #{y}” }
    C.new.instance_exec(@y) { |y| “@x: #{@x}, @y: #{y}” }
  end
end
#D.new.twisted_method  # => “@x: 1, @y: ”
D.new.twisted_method  # => “@x: 1, @y: 2”

因為調(diào)用instance_eval后,將調(diào)用者作為了當(dāng)前的self,所以作用域更換到了class C中,之前的作用域就不生效了。這時如果還想訪問到之前@y變量,就需要通過參數(shù)打包上@y一起隨instance_eval轉(zhuǎn)義,但因為instance_eval不能攜帶參數(shù),所以使用其同胞兄弟instance_exec方法。


instance_eval 與 class_eval 的區(qū)別
###instance_eval
首先從名字可以得到的信息是,instance_eval的調(diào)用者receiver必須是一個實例instance,而在instance_eval block的內(nèi)部,self即為receiver實例本身。

obj_instance.instance_eval do
 self # => obj_instance
 # current class => obj_instance's singleton class
end
!--more-->

根據(jù)這個定義,如果在一個實例上調(diào)用了instance_eval,就可以在其中定義該實例的單態(tài)函數(shù) singleton_method

class A
end

a = A.new
a.instance_eval do
 self # => a
 # current class => a's singleton class
 def method1
  puts 'this is a singleton method of instance a'
 end
end

a.method1
#=> this is a singleton method of instance a

b = A.new
b.method1
#=>NoMethodError: undefined method `method1' for #A:0x10043ff70>

同樣,因為類class本身也是Class類的一個實例,instance_eval也可以用在類上,這個時候就可以在其中定義該類的singleton_method,即為該類的類函數(shù)。

換句話說,可以用instance_eval來定義類函數(shù)class method,這比較容易混淆,需要搞清楚。

class A
end

A.instance_eval do
 self # => A
 # current class => A's singleton class
 def method1
  puts 'this is a singleton method of class A'
 end
end

A.method1
#=> this is a singleton method of class A
class_eval

###class_eval

再來看class_eval,首先從名字可以得到的信息是,class_eval的調(diào)用者receiver必須是一個類,而在class_eval block的內(nèi)部,self即為receiver類本身。

class A
end

A.class_eval do
 self # => A
 # current class => A
end

根據(jù)這個定義,如果在一個類上調(diào)用了class_eval,就可以在其中定義該類的實例函數(shù) instance_method

class A
end

a = A.new
a.method1
#=> NoMethodError: undefined method `method1' for #A:0x10043ff70>

A.class_eval do
 self # => A
 # current class => A
 def method1
  puts 'this is a instance method of class A'
 end
end

a.method1
#=> this is a instance method of class A

換句話說,可以用class_eval來定義實例函數(shù)instance method,這也比較容易混淆,需要搞清楚。

您可能感興趣的文章:
  • Ruby 字符串處理
  • Ruby字符串、條件、循環(huán)、數(shù)組、Hash、類基本操作筆記

標(biāo)簽:安康 山西 海南 崇左 山西 長沙 喀什 濟南

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《詳解Ruby中的instance_eval方法及其與class_eval的對比》,本文關(guān)鍵詞  ;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266
    夏津县| 德清县| 二连浩特市| 昌黎县| 嵊州市| 建瓯市| 太和县| 嘉义市| 建始县| 德州市| 开江县| 滦南县| 江阴市| 武穴市| 腾冲县| 外汇| 建阳市| 蓝山县| 广河县| 紫阳县| 石渠县| 岳普湖县| 佛学| 西宁市| 明星| 齐河县| 额尔古纳市| 武乡县| 黔西县| 小金县| 蚌埠市| 石家庄市| 班戈县| 鹰潭市| 黔西县| 乌什县| 万安县| 梁河县| 张掖市| 略阳县| 阿拉善右旗|