ruby中的反射(reflection)應(yīng)用實(shí)例

字號(hào):


    這篇文章主要介紹了ruby中的反射(reflection)應(yīng)用實(shí)例,實(shí)現(xiàn)通過(guò)一個(gè)類名字符串構(gòu)造一個(gè)類對(duì)象和訪問(wèn)成員變量和私有方法,需要的朋友可以參考下。
    在java語(yǔ)言中,提供了發(fā)射機(jī)制,通過(guò)發(fā)射機(jī)制可以通過(guò)字符串構(gòu)造出這個(gè)對(duì)象,可以獲取對(duì)象的所有方法(包括私有方法),可以調(diào)用私有方法,可以更改成員變量的值(包括私有的成員變量)。
    ruby也是面向?qū)ο蟮母呒?jí)語(yǔ)言,當(dāng)然也提供了反射機(jī)制,今天我們討論通過(guò)類名稱構(gòu)造類對(duì)象的功能。
    一、通過(guò)類名稱構(gòu)造類對(duì)象
    我們先看普通的構(gòu)造:
    代碼如下:
    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
    定義了一個(gè)基礎(chǔ)類wood,有兩個(gè)子類:wooddesk,woodchair,子類有分別有一個(gè)私有方法say_private。
    我們new出對(duì)象來(lái)執(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í)行,返回結(jié)果是不能執(zhí)行,desk.respond_to?:say_private返回false:
    代碼如下:
    iamaprimalwood
    iamadeskmadeofwood
    iamachairmadeofwood
    deskrespondtosay_private?false
    好,現(xiàn)在我們通過(guò)反射機(jī)制來(lái)構(gòu)造對(duì)象,并嘗試執(zhí)行其私有方法。
    我們注意到模塊的定義中有三個(gè)常量,定義的是類名稱,
    代碼如下:
    #theclassname,laterwewilluseittocreatethecorrespondingobject
    class_name_of_wood=modulea::wood
    class_name_of_wooddesk=modulea::wooddesk
    class_name_of_woodchair=modulea::woodchair
    下面會(huì)通過(guò)這三個(gè)變量來(lái)理解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,這個(gè)方法是module模塊中的,其作用是返回模塊中所有常量的symbol對(duì)象。我們看結(jié)果輸出:
    代碼如下:
    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
    從結(jié)果中看到,定義的三個(gè)常量和類名稱都被返回了。所以注意:ruby中的常量是包含定義的常量(變量)和類名稱,注意他們都是symbol對(duì)象。。
    不過(guò)我們是需要根據(jù)類名稱構(gòu)造類對(duì)象,那么那三個(gè)常量就是沒用的,需要?jiǎng)h除。我們通過(guò)正則表達(dá)式匹配名字,來(lái)過(guò)濾。上面的代碼修改一下:
    代碼如下:
    #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,這樣我們就過(guò)濾掉了那三個(gè)常量。
    找都類名稱之后,開始構(gòu)造對(duì)象:
    代碼如下:
    #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
    調(diào)用symbol的new方法構(gòu)造出次對(duì)象(sym.new),然后我們調(diào)用對(duì)象的say方法:
    代碼如下:
    createtheobject:#
    createtheobject:#
    createtheobject:#
    iamaprimalwood
    iamadeskmadeofwood
    iamachairmadeofwood
    達(dá)到了我們預(yù)期的結(jié)果。
    二、操作成員變量和私有方法
    使用過(guò)java反射的同學(xué)們都知道,有了對(duì)象之后,操作成員變量和私有方法也就不在話下了。
    ruby中也是一樣。
    先看操作成員變量的例子。我們嘗試更改一個(gè)成員變量的值。(接著上一片文章的代碼)
    代碼如下:
    #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,我們得到一個(gè)wood對(duì)象,然后調(diào)用其instance_variables方法得到所有成員變量的名稱(symbol對(duì)象)。
    2、然后,調(diào)用對(duì)象的first_wood.instance_variable_get方法,傳遞成員變量名稱,得到成員變量對(duì)象。
    3、最后,我們通過(guò)first_wood.instance_variable_set,改變這個(gè)成員變量的值。
    代碼運(yùn)行結(jié)果:
    代碼如下:
    classofvar=symbol,valueofvar=@desc
    classofvar_value=string,valueofvar_value=iamaprimalwood
    iamaprimalwood...andiwaschanged.
    再看調(diào)用私有方法:
    代碼如下:
    #callprivatemethod
    last_wood=obj_list.last
    last_wood.method(:say_private).call
    很簡(jiǎn)單,如果你知道方法名稱,調(diào)用last_wood.method傳入方法名,就可以得到一個(gè)method對(duì)象,然后調(diào)用method對(duì)象的call方法,結(jié)果是私有方法輸出的內(nèi)容:
    代碼如下:
    iwantgetmarriedwithawooddesk...
    普通場(chǎng)景下用不到修改成員變量和調(diào)用私有方法,因?yàn)檫@是違反了面向?qū)ο蟮姆庋b原則的,那么反射在什么場(chǎng)景下有用呢?從我個(gè)人經(jīng)驗(yàn)來(lái)說(shuō)我覺得兩個(gè)地方有用:
    1)單元測(cè)試。
    2)面向方面編程。
    這兩種場(chǎng)景都需要調(diào)用私有方法或替換成員變量的值。