ruby中的反射(reflection)應用實例
來源:易賢網(wǎng) 閱讀:727 次 日期:2016-06-17 09:34:06
溫馨提示:易賢網(wǎng)小編為您整理了“ruby中的反射(reflection)應用實例”,方便廣大網(wǎng)友查閱!

這篇文章主要介紹了ruby中的反射(reflection)應用實例,實現(xiàn)通過一個類名字符串構造一個類對象和訪問成員變量和私有方法,需要的朋友可以參考下。

在java語言中,提供了發(fā)射機制,通過發(fā)射機制可以通過字符串構造出這個對象,可以獲取對象的所有方法(包括私有方法),可以調用私有方法,可以更改成員變量的值(包括私有的成員變量)。

ruby也是面向對象的高級語言,當然也提供了反射機制,今天我們討論通過類名稱構造類對象的功能。

一、通過類名稱構造類對象

我們先看普通的構造:

代碼如下:

modulemodulea

#theclassname,laterwewilluseittocreatethecorrespondingobject

class_name_of_wood=modulea::wood

class_name_of_wooddesk=modulea::wooddesk

class_name_of_woodchair=modulea::woodchair

classwood

definitialize

@desc=iamaprimalwood

end

defsay

puts@desc

end

end

classwooddesk<wood

definitialize

@desc=iamadeskmadeofwood

end

defsay_private

putsactually,ihavesomebugbutnopublic

end

public:say

private:say_private

end

classwoodchair<wood

definitialize

@desc=iamachairmadeofwood

end

defsay_private

putsiwantgetmarriedwithawooddesk...

end

defsmile

putshahahhahhaha....

end

public:say

private:say_private,:smile

end

end

定義了一個基礎類wood,有兩個子類:wooddesk,woodchair,子類有分別有一個私有方法say_private。

我們new出對象來執(zhí)行:

代碼如下:

#thenormalinitailze

wood=modulea::wood.new

wood.say

desk=modulea::wooddesk.new

desk.say

chair=modulea::woodchair.new

chair.say

#trycalltheprivatemethod

putsdeskrespondtosay_private?#{desk.respond_to?:say_private}

desk.say_privateifdesk.respond_to?:say_private

上面代碼,執(zhí)行public方法say,然后嘗試執(zhí)行private方法say_private,執(zhí)行先check是否能夠執(zhí)行,返回結果是不能執(zhí)行,desk.respond_to?:say_private返回false:

代碼如下:

iamaprimalwood

iamadeskmadeofwood

iamachairmadeofwood

deskrespondtosay_private?false

好,現(xiàn)在我們通過反射機制來構造對象,并嘗試執(zhí)行其私有方法。

我們注意到模塊的定義中有三個常量,定義的是類名稱,

代碼如下:

#theclassname,laterwewilluseittocreatethecorrespondingobject

class_name_of_wood=modulea::wood

class_name_of_wooddesk=modulea::wooddesk

class_name_of_woodchair=modulea::woodchair

下面會通過這三個變量來理解module.constants方法。

下面代碼片段,基于上面的類定義:

代碼如下:

#getallmoduleconstants

obj_list=array.new

tmp_const_sym_list=modulea.constants

tmp_const_sym_list.eachdo|sym|

obj_list<<modulea.const_get(sym)

putscalss=#{sym.class},value=#{sym}

end

我們注意到modulea.constants,這個方法是module模塊中的,其作用是返回模塊中所有常量的symbol對象。我們看結果輸出:

代碼如下:

calss=symbol,value=class_name_of_wood

calss=symbol,value=class_name_of_wooddesk

calss=symbol,value=class_name_of_woodchair

calss=symbol,value=wood

calss=symbol,value=wooddesk

calss=symbol,value=woodchair

從結果中看到,定義的三個常量和類名稱都被返回了。所以注意:ruby中的常量是包含定義的常量(變量)和類名稱,注意他們都是symbol對象。。

不過我們是需要根據(jù)類名稱構造類對象,那么那三個常量就是沒用的,需要刪除。我們通過正則表達式匹配名字,來過濾。上面的代碼修改一下:

代碼如下:

#getallmoduleconstants

sym_list=array.new

tmp_const_sym_list=modulea.constants

tmp_const_sym_list.eachdo|sym|

putscalss=#{sym.class},value=#{sym}

sym_list<<modulea.const_get(sym)if/^wood\w*/=~sym.to_s

end

sym_list<<modulea.const_get(sym)if/^wood\w*/=~sym.to_s,僅保存以wood開頭的symbol,這樣我們就過濾掉了那三個常量。

找都類名稱之后,開始構造對象:

代碼如下:

#createobjectfromsymbol

obj_list=array.new

sym_list.eachdo|sym|

obj=sym.new

obj_list<<obj

putscreatetheobject:#{obj}

end

begin

obj_list.eachdo|wood|

wood.say

end

調用symbol的new方法構造出次對象(sym.new),然后我們調用對象的say方法:

代碼如下:

createtheobject:#

createtheobject:#

createtheobject:#

iamaprimalwood

iamadeskmadeofwood

iamachairmadeofwood

達到了我們預期的結果。

二、操作成員變量和私有方法

使用過java反射的同學們都知道,有了對象之后,操作成員變量和私有方法也就不在話下了。

ruby中也是一樣。

先看操作成員變量的例子。我們嘗試更改一個成員變量的值。(接著上一片文章的代碼)

代碼如下:

#manpulateinstancevariables

first_wood=obj_list.first

first_wood.instance_variables.eachdo|var|

#gettheinstancevariable

putsclassofvar=#{var.class},valueofvar=#{var}

var_value=first_wood.instance_variable_get(var)

putsclassofvar_value=#{var_value.class},valueofvar_value=#{var_value}

#setthenewvalueofinstancevarialbe

first_wood.instance_variable_set(var,var_value+...andiwaschanged.)

first_wood.say

end

1、first_wood.instance_variables.each,我們得到一個wood對象,然后調用其instance_variables方法得到所有成員變量的名稱(symbol對象)。

2、然后,調用對象的first_wood.instance_variable_get方法,傳遞成員變量名稱,得到成員變量對象。

3、最后,我們通過first_wood.instance_variable_set,改變這個成員變量的值。

代碼運行結果:

代碼如下:

classofvar=symbol,valueofvar=@desc

classofvar_value=string,valueofvar_value=iamaprimalwood

iamaprimalwood...andiwaschanged.

再看調用私有方法:

代碼如下:

#callprivatemethod

last_wood=obj_list.last

last_wood.method(:say_private).call

很簡單,如果你知道方法名稱,調用last_wood.method傳入方法名,就可以得到一個method對象,然后調用method對象的call方法,結果是私有方法輸出的內容:

代碼如下:

iwantgetmarriedwithawooddesk...

普通場景下用不到修改成員變量和調用私有方法,因為這是違反了面向對象的封裝原則的,那么反射在什么場景下有用呢?從我個人經(jīng)驗來說我覺得兩個地方有用:

1)單元測試。

2)面向方面編程。

這兩種場景都需要調用私有方法或替換成員變量的值。

更多信息請查看腳本欄目
易賢網(wǎng)手機網(wǎng)站地址:ruby中的反射(reflection)應用實例

2025國考·省考課程試聽報名

  • 報班類型
  • 姓名
  • 手機號
  • 驗證碼
關于我們 | 聯(lián)系我們 | 人才招聘 | 網(wǎng)站聲明 | 網(wǎng)站幫助 | 非正式的簡要咨詢 | 簡要咨詢須知 | 加入群交流 | 手機站點 | 投訴建議
工業(yè)和信息化部備案號:滇ICP備2023014141號-1 云南省教育廳備案號:云教ICP備0901021 滇公網(wǎng)安備53010202001879號 人力資源服務許可證:(云)人服證字(2023)第0102001523號
聯(lián)系電話:0871-65099533/13759567129 獲取招聘考試信息及咨詢關注公眾號:hfpxwx
咨詢QQ:526150442(9:00—18:00)版權所有:易賢網(wǎng)