簡(jiǎn)便方法的用法
現(xiàn)有一個(gè)字符串列表,需要對(duì)其中的每個(gè)字符串執(zhí)行轉(zhuǎn)換大寫(xiě)操作,我們可以用一個(gè)簡(jiǎn)便寫(xiě)法來(lái)完成。
復(fù)制代碼 代碼如下:
name_list = ["chareice", "angel"]
name_list.map(:upcase)
# => ["CHAREICE", "ANGEL"]
這個(gè)寫(xiě)法等同于
復(fù)制代碼 代碼如下:
name_list.map do {|name| name.upcase}
簡(jiǎn)便寫(xiě)法帶來(lái)的是很明顯的效率提升,可是這看似魔術(shù)一般的參數(shù),背后的原理是怎樣的呢?
符號(hào)
如果把上面方法調(diào)用的符號(hào)去掉,可以很明顯得看到,是把:upcase這個(gè)符號(hào)傳到方法中,作為方法的參數(shù)。
實(shí)際上,符號(hào)代表的是塊轉(zhuǎn)變?yōu)镻roc(block-to-proc conversion)。我們看下面的一個(gè)例子。
復(fù)制代碼 代碼如下:
def capture_block(block)
block.call
end
capture_block { puts "我有一只小毛驢,我從來(lái)也不騎。" }
# => 我有一只小毛驢,我從來(lái)也不騎。
我們運(yùn)行capture_block函數(shù),給它傳遞一個(gè)代碼塊,代碼塊會(huì)經(jīng)符號(hào)的轉(zhuǎn)換變?yōu)橐粋€(gè)Proc對(duì)象傳遞到函數(shù)中,在上面的例子中就是block變量。如果我們輸出一下block的class,輸出的結(jié)果會(huì)是Proc。
你也可以將一個(gè)Proc對(duì)象傳遞給capture_block來(lái)代替代碼塊.
復(fù)制代碼 代碼如下:
p = Proc.new { puts "又給一只小毛驢" }
capture_block(p)
# => 又給一只小毛驢
這里看來(lái)符號(hào)是多余的,完全可以去掉,運(yùn)行的結(jié)果也是一樣。
符號(hào)做了什么?
以capture_block(p)調(diào)用為例。
1.觸發(fā)p的to_proc方法。
2.告訴Ruby解釋器,將to_proc方法返回的結(jié)果當(dāng)做本次函數(shù)調(diào)用的block。
如果同時(shí)使用了符號(hào)和傳入了block給一個(gè)函數(shù),Ruby會(huì)報(bào)錯(cuò)。
復(fù)制代碼 代碼如下:
capture_block(p) { puts "傳給一個(gè)block" }
#=>SyntaxError: (irb):30: both block arg and actual block given
所以將一個(gè)Proc對(duì)象傳給符號(hào),它會(huì)調(diào)用Proc對(duì)象的to_proc方法,返回它自己,然后把它當(dāng)做方法調(diào)用的block傳遞給方法。
:upcase是什么?
知道了符號(hào)的作用后,我們可以看到,:upcase是先調(diào)用了:upcase對(duì)象的to_proc方法。
:upcase的to_proc方法實(shí)現(xiàn)如下:
復(fù)制代碼 代碼如下:
class Symbol
def to_proc
Proc.new {|obj| obj.send(self) }
end
end
這下結(jié)果就很清楚了,Symbol#to_proc會(huì)返回一個(gè)帶參數(shù)的Proc對(duì)象,Proc對(duì)象所做的是為使用這個(gè)Proc對(duì)象的對(duì)象發(fā)送調(diào)用名字為該符號(hào)的方法。
您可能感興趣的文章:- Ruby Gems更換淘寶源方法
- Ruby中訪問(wèn)SQL Server數(shù)據(jù)庫(kù)的配置實(shí)例
- 舉例理解Ruby on Rails的頁(yè)面緩存機(jī)制