Ruby的塊block是ruby的一個重要特性,它能夠允許用戶將一系列的自定義操作(說白了就是一坨代碼)像參數(shù)一樣傳遞給方法,接受block塊的方法會決定執(zhí)行塊的方式以及次數(shù),在ruby中最常見使用塊的地方也就是一些循環(huán)的方法,將循環(huán)的元素依次用塊的邏輯執(zhí)行一遍,當然具體情況大同小異,你也能自定義自己的能夠接受塊的方法。
提升代碼抽象能力
先來看個簡單的例子:
[1, 2, 3, 4, 5].each do |i|
puts i ** 2
end
例子中的意思很明顯,循環(huán)數(shù)組中的每個數(shù)字,每次打印這個數(shù)字的平方數(shù)到標準輸出界面。
each方法只是提供一個遍歷的方式,具體要執(zhí)行的邏輯由用戶使用的時候具體制定,這其實又一點模版方法的意味在里面,抽象出來的模版方法只提供基礎(chǔ)的骨架,具體的方式在使用的時候后綁定,達到一定抽象的目的,當然咯這里只是類比,each返回的其實是一個Enumerator,迭代每個數(shù)據(jù)的時候會執(zhí)行塊的邏輯。
所以理解塊的作用的時候第一個重點應該放在提升代碼抽象能力上面,而且避免咯我們把代碼寫的非常過程化。
代碼復用
如標題,block的方式其實也能很好的提供代碼復用的能力,同樣依例子開始:
>> [1, "5", 2, "7", "3"].sort_by { |a| a.to_i }
=> [1, 2, "3", "5", "7"]
對一個包含數(shù)組和字符兩種內(nèi)容的混合數(shù)組排序,排序的邏輯是把它們都轉(zhuǎn)換成數(shù)字之后進行比較。倘若沒有塊,我們在實現(xiàn)的時候只能手動循環(huán)遍歷和轉(zhuǎn)換比較,但這樣也就產(chǎn)生一個問題,十種不同方式的排序我們就得實現(xiàn)十次差不多的邏輯(因為只有比較的方式那一行不一樣而已),代碼會非常冗余和難以維護。
數(shù)組從Enumerable模塊那mixin的所有方法包括上面例子中的sort_by方法把一些常用的邏輯都包含在里面了,用戶在消費集合或者自定義的數(shù)據(jù)類型的時候通過mixin的方式獲得這些抽象處理的方法之后使用的時候能夠方便注入自己的邏輯達到想要的效果。
所以代碼度用絕對算得上是有塊之后一個優(yōu)雅且實用的功能
自定義帶塊的方法
自定義一個能夠接受使用塊的方法非常簡單,只需要借助于yield在合理的地方中斷和交替執(zhí)行方法和塊的邏輯,而且yield也能很方便的在兩者之間傳遞數(shù)據(jù)。
yield其實并不是一個多么神奇的東西,說直白點就是暫停當前的邏輯然后轉(zhuǎn)到你設(shè)置的其他地方繼續(xù)執(zhí)行,next調(diào)用的時候再次回到先前暫停的地方,最常用的是用來實現(xiàn)無窮迭代(因為只保存當前信息/狀態(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)于塊的第一部分介紹差不多就是這樣,剩下的一些細節(jié),比如塊的控制,塊局部變量,如何將塊封裝成Proc對象,接受Proc對象的時候,自定義方法使用封裝成Proc對象的塊的時候,實現(xiàn)的一些區(qū)別,以及Proc帶來的像是復用塊的邏輯的一些好處吧。