一,塊的聲明
塊的聲明在函數(shù)調(diào)用之后,用{..}括起來(lái),或do..end封裝。{}一般用在單行語(yǔ)句上,do..end用在多行語(yǔ)句上。
(1..4).each{|v| print "#{v} "} #輸出1 2 3 4
塊可以帶參數(shù),與函數(shù)參數(shù)不同,塊參數(shù)用||封裝,當(dāng)然,可以帶多個(gè)參數(shù)。這些參數(shù)怎么定義,實(shí)際上是在函數(shù)內(nèi)部定義好的,后面會(huì)講到。
二,塊內(nèi)變量的訪問(wèn)
塊內(nèi)可以訪問(wèn)塊外的變量,也就是塊外的變量在塊內(nèi)是可見(jiàn)的,如
sum = 0
(1..5).each do |v|
name = 'smile' #name屬于塊內(nèi)變量,其可視范圍只能在塊內(nèi)。假設(shè)塊外沒(méi)有相同名稱(chēng)的變量.
sum += v #sum在塊內(nèi)可見(jiàn)
end
p sum #輸出15,sum已改變。
p name #Error! name不可訪問(wèn)。
正因塊內(nèi)可以塊外的變量所以可能不小心修改了一些外部變量,這是我們不希望的。幸運(yùn)的是Ruby1.9版本后,提供了一種安全的方式聲明塊內(nèi)變量,在塊參數(shù)后面加";",塊內(nèi)變量放在";"之后.
name = 'outside'
sum = 0
(1..5).each do |v;name| #name在";"之后,可以聲明多個(gè)變量,用逗號(hào)隔開(kāi)
name = 'inside' #name屬于塊內(nèi)變量,其可視范圍只能在塊內(nèi).假設(shè)塊外沒(méi)有相同名稱(chēng)的變量。
sum += v #sum在塊內(nèi)可訪問(wèn)
end
p sum #輸出15,sum已改變。
p name #輸出outside,沒(méi)有變。
三,yield語(yǔ)句
看這里,可能還不是很明白,函數(shù)是如何調(diào)用塊的?,F(xiàn)在就來(lái)介紹塊的調(diào)用,關(guān)鍵是yield語(yǔ)句。在函數(shù)體中,如果用yield,函數(shù)會(huì)調(diào)用函數(shù)的塊。
def threeTime
yield
yield
yield
end
threeTime{p 'Hello world!'}
輸出三行Hello world!,是不是很簡(jiǎn)單呢。現(xiàn)在應(yīng)該明白了吧,是yield調(diào)用的塊。
塊的參數(shù)是怎么回事呢?估計(jì)你已經(jīng)想到了,就是yield的參數(shù),跟一般函數(shù)一樣yield可以帶參數(shù)的??蠢?/p>
def takeBlock(p1)
if block_given? # 判斷是否有塊,如果在yield時(shí),沒(méi)有聲明塊,會(huì)出錯(cuò),所以在這里作判斷會(huì)好點(diǎn)。
yield(p1) #把p1傳給塊參數(shù),既下面塊聲明中的s
else
p1
end
endie
takeBlock("no block") #輸出"no block"
takeBlock("no block") { |s| s.sub(/no /, '') } #輸出"block"
既然yield能傳參數(shù)給塊,反過(guò)來(lái),塊能不能傳值給yield呢?答案是肯定的。塊中最后一句語(yǔ)句的值會(huì)自動(dòng)傳給yield。請(qǐng)看示例
def nTime
i = yield #第一次調(diào)用時(shí),返回塊的值
(0..i).each {|v| yield(v)} # 此處yield也可以放在塊中
end
nTime do |v|
print "#{v} " if v
9 #yield調(diào)用時(shí)返回的數(shù)
end
#輸出1 2 3 4 5 6 7 8 9
當(dāng)然上例只是拿來(lái)做例子,實(shí)際上沒(méi)有人會(huì)這樣定義,更好的定義如下:
def nTime(n)
(0..n).each {|v| yield(v)}
end
nTime(9) do |v|
print "#{v} "
end
我們來(lái)看下Array中的find實(shí)現(xiàn)
class Array
def find
for i in 0...size
value = self[i]
return value if yield(value)
end
return nil
end
end
[1, 3, 5, 7, 9].find {|v| v > 5 } #實(shí)現(xiàn)查找第一個(gè)大于5的數(shù),輸出7。
因?yàn)閴K的出現(xiàn),Ruby中少了許多for語(yǔ)句,代碼看上去更人性化,寫(xiě)代碼不再是枯燥的事,而是一種享受。
四,傳遞塊的另一種方式
def fun #不帶參數(shù)的
yield
end
proc = ->{p 'haha'}
fun proc
#####
def fun2(x) #帶參數(shù)的
yield x
end
proc2 = ->(x){p x}
fun2 1,proc2
五,instance_eval()和instance_exec()
在Ruby中,提供了一個(gè)非??岬奶匦裕梢酝ㄟ^(guò)使用Objec#instance_eval(), Objec#instance_exec()方法插入一個(gè)代碼塊,做一個(gè)的對(duì)象上下文探針(Context Proble),深入到對(duì)象中的代碼片段,對(duì)其進(jìn)行操作。有了這個(gè)特性以后,就可以很輕松的測(cè)試對(duì)象的行為,查看對(duì)象的當(dāng)前狀態(tài)。
class MyClass
def initialize
@v = 1;
end
end
obj = MyClass.new
obj.instance_eval do
puts self # => #MyClass:0x007fbb2d0299b0>
puts @v # => 1
end
obj.instance_exec(5) { |x| puts x * @v } # => 5
您可能感興趣的文章:- 深入理解Ruby中的代碼塊block特性
- Ruby中的block代碼塊學(xué)習(xí)教程
- 詳解Ruby中的代碼塊對(duì)象Proc