欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

ruby迭代map的簡潔寫法實(shí)現(xiàn)原理分析

 更新時間:2014年11月05日 10:52:55   投稿:junjie  
這篇文章主要介紹了ruby迭代map的簡潔寫法實(shí)現(xiàn)原理分析,本文著重對簡潔寫法的原理進(jìn)行解析,需要的朋友可以參考下

簡便方法的用法

現(xiàn)有一個字符串列表,需要對其中的每個字符串執(zhí)行轉(zhuǎn)換大寫操作,我們可以用一個簡便寫法來完成。

復(fù)制代碼 代碼如下:

name_list = ["chareice", "angel"]
name_list.map(&:upcase)
# => ["CHAREICE", "ANGEL"]

這個寫法等同于
復(fù)制代碼 代碼如下:

name_list.map do {|name| name.upcase}

簡便寫法帶來的是很明顯的效率提升,可是這看似魔術(shù)一般的參數(shù),背后的原理是怎樣的呢?

&符號

如果把上面方法調(diào)用的&符號去掉,可以很明顯得看到,是把:upcase這個符號傳到方法中,作為方法的參數(shù)。

實(shí)際上,&符號代表的是塊轉(zhuǎn)變?yōu)镻roc(block-to-proc conversion)。我們看下面的一個例子。

復(fù)制代碼 代碼如下:

def capture_block(&block)
  block.call
end

capture_block { puts "我有一只小毛驢,我從來也不騎。" }
# => 我有一只小毛驢,我從來也不騎。

我們運(yùn)行capture_block函數(shù),給它傳遞一個代碼塊,代碼塊會經(jīng)&符號的轉(zhuǎn)換變?yōu)橐粋€Proc對象傳遞到函數(shù)中,在上面的例子中就是block變量。如果我們輸出一下block的class,輸出的結(jié)果會是Proc。

你也可以將一個Proc對象傳遞給capture_block來代替代碼塊.

復(fù)制代碼 代碼如下:

p = Proc.new { puts "又給一只小毛驢" }
capture_block(&p)
# => 又給一只小毛驢

這里看來&符號是多余的,完全可以去掉&,運(yùn)行的結(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會報(bào)錯。

復(fù)制代碼 代碼如下:

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方法實(shí)現(xiàn)如下:

復(fù)制代碼 代碼如下:

class Symbol
  def to_proc
    Proc.new {|obj| obj.send(self) }
  end
end

這下結(jié)果就很清楚了,Symbol#to_proc會返回一個帶參數(shù)的Proc對象,Proc對象所做的是為使用這個Proc對象的對象發(fā)送調(diào)用名字為該符號的方法。

相關(guān)文章

最新評論