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