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