Ruby的塊block是ruby的一個重要特性虐沥,它能夠允許用戶將一系列的自定義操作(說白了就是一坨代碼)像參數(shù)一樣傳遞給方法钮糖,接受block塊的方法會決定執(zhí)行塊的方式以及次數(shù)史简,在ruby中最常見使用塊的地方也就是一些循環(huán)的方法鲸匿,將循環(huán)的元素依次用塊的邏輯執(zhí)行一遍回俐,當(dāng)然具體情況大同小異,你也能自定義自己的能夠接受塊的方法惰蜜。
提升代碼抽象能力
先來看個簡單的例子:
[1, 2, 3, 4, 5].each do |i|
puts i ** 2
end
例子中的意思很明顯,循環(huán)數(shù)組中的每個數(shù)字受神,每次打印這個數(shù)字的平方數(shù)到標(biāo)準(zhǔn)輸出界面抛猖。
each方法只是提供一個遍歷的方式,具體要執(zhí)行的邏輯由用戶使用的時候具體制定鼻听,這其實(shí)又一點(diǎn)模版方法的意味在里面财著,抽象出來的模版方法只提供基礎(chǔ)的骨架,具體的方式在使用的時候后綁定撑碴,達(dá)到一定抽象的目的撑教,當(dāng)然咯這里只是類比,each返回的其實(shí)是一個Enumerator醉拓,迭代每個數(shù)據(jù)的時候會執(zhí)行塊的邏輯伟姐。
所以理解塊的作用的時候第一個重點(diǎn)應(yīng)該放在提升代碼抽象能力上面,而且避免咯我們把代碼寫的非常過程化亿卤。
代碼復(fù)用
如標(biāo)題愤兵,block的方式其實(shí)也能很好的提供代碼復(fù)用的能力,同樣依例子開始:
>> [1, "5", 2, "7", "3"].sort_by { |a| a.to_i }
=> [1, 2, "3", "5", "7"]
對一個包含數(shù)組和字符兩種內(nèi)容的混合數(shù)組排序排吴,排序的邏輯是把它們都轉(zhuǎn)換成數(shù)字之后進(jìn)行比較秆乳。倘若沒有塊,我們在實(shí)現(xiàn)的時候只能手動循環(huán)遍歷和轉(zhuǎn)換比較傍念,但這樣也就產(chǎn)生一個問題矫夷,十種不同方式的排序我們就得實(shí)現(xiàn)十次差不多的邏輯(因?yàn)橹挥斜容^的方式那一行不一樣而已)葛闷,代碼會非常冗余和難以維護(hù)。
數(shù)組從Enumerable模塊那mixin的所有方法包括上面例子中的sort_by方法把一些常用的邏輯都包含在里面了双藕,用戶在消費(fèi)集合或者自定義的數(shù)據(jù)類型的時候通過mixin的方式獲得這些抽象處理的方法之后使用的時候能夠方便注入自己的邏輯達(dá)到想要的效果淑趾。
所以代碼度用絕對算得上是有塊之后一個優(yōu)雅且實(shí)用的功能
自定義帶塊的方法
自定義一個能夠接受使用塊的方法非常簡單,只需要借助于yield在合理的地方中斷和交替執(zhí)行方法和塊的邏輯忧陪,而且yield也能很方便的在兩者之間傳遞數(shù)據(jù)扣泊。
yield其實(shí)并不是一個多么神奇的東西,說直白點(diǎn)就是暫停當(dāng)前的邏輯然后轉(zhuǎn)到你設(shè)置的其他地方繼續(xù)執(zhí)行嘶摊,next調(diào)用的時候再次回到先前暫停的地方延蟹,最常用的是用來實(shí)現(xiàn)無窮迭代(因?yàn)橹槐4娈?dāng)前信息/狀態(tài)),其它高級的功能得看具體的語言叶堆。
def total(from, to)
result = 0
from.upto(to) do |num|
if block_given?
result += yield(num)
else
result += num
end
end
return result
end
p total(1, 10) #=> 55
p total(1, 10){|num| num ** 2 } #=> 385
關(guān)于塊的第一部分介紹差不多就是這樣阱飘,剩下的一些細(xì)節(jié),比如塊的控制虱颗,塊局部變量沥匈,如何將塊封裝成Proc對象,接受Proc對象的時候忘渔,自定義方法使用封裝成Proc對象的塊的時候高帖,實(shí)現(xiàn)的一些區(qū)別,以及Proc帶來的像是復(fù)用塊的邏輯的一些好處吧畦粮。