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