在Ruby中,代碼塊不是一個對象,但它可以被proc替換為一個對象。接下來,讓武林技術(shù)頻道小編詳解Ruby中的代碼塊及其參數(shù)傳遞,希望對你學(xué)習(xí)這方面知識有幫助!
一,塊的聲明?
? 塊的聲明在函數(shù)調(diào)用之后,用{..}括起來,或do..end封裝。{}一般用在單行語句上,do..end用在多行語句上。
ruby;">(1..4).each{|v| print "#{v} "} #輸出1 2 3 4
? 塊可以帶參數(shù),與函數(shù)參數(shù)不同,塊參數(shù)用||封裝,當(dāng)然,可以帶多個參數(shù)。這些參數(shù)怎么定義,實際上是在函數(shù)內(nèi)部定義好的,后面會講到。
二,塊內(nèi)變量的訪問?
? 塊內(nèi)可以訪問塊外的變量,也就是塊外的變量在塊內(nèi)是可見的,如
sum = 0 (1..5).each do |v| name = 'smile' #name屬于塊內(nèi)變量,其可視范圍只能在塊內(nèi)。假設(shè)塊外沒有相同名稱的變量. sum += v #sum在塊內(nèi)可見 end p sum #輸出15,sum已改變。 p name #Error! name不可訪問。
? 正因塊內(nèi)可以塊外的變量所以可能不小心修改了一些外部變量,這是我們不希望的。幸運的是Ruby1.9版本后,提供了一種安全的方式聲明塊內(nèi)變量,在塊參數(shù)后面加";",塊內(nèi)變量放在";"之后.
name = 'outside' sum = 0 (1..5).each do |v;name| #name在";"之后,可以聲明多個變量,用逗號隔開 name = 'inside' #name屬于塊內(nèi)變量,其可視范圍只能在塊內(nèi).假設(shè)塊外沒有相同名稱的變量。 sum += v #sum在塊內(nèi)可訪問 end p sum #輸出15,sum已改變。 p name #輸出outside,沒有變。
三,yield語句?
? 看這里,可能還不是很明白,函數(shù)是如何調(diào)用塊的。現(xiàn)在就來介紹塊的調(diào)用,關(guān)鍵是yield語句。在函數(shù)體中,如果用yield,函數(shù)會調(diào)用函數(shù)的塊。
def threeTime yield yield yield end threeTime{p 'Hello world!'}
? 輸出三行Hello world!,是不是很簡單呢。現(xiàn)在應(yīng)該明白了吧,是yield調(diào)用的塊。
塊的參數(shù)是怎么回事呢?估計你已經(jīng)想到了,就是yield的參數(shù),跟一般函數(shù)一樣yield可以帶參數(shù)的。看例子
def takeBlock(p1) if block_given? # 判斷是否有塊,如果在yield時,沒有聲明塊,會出錯,所以在這里作判斷會好點。 yield(p1) #把p1傳給塊參數(shù),既下面塊聲明中的s else p1 end endie takeBlock("no block") #輸出"no block" takeBlock("no block") { |s| s.sub(/no /, '') } #輸出"block"
? 既然yield能傳參數(shù)給塊,反過來,塊能不能傳值給yield呢?答案是肯定的。塊中最后一句語句的值會自動傳給yield。請看示例
def nTime i = yield #第一次調(diào)用時,返回塊的值 (0..i).each {|v| yield(v)} # 此處yield也可以放在塊中 end nTime do |v| print "#{v} " if v 9 #yield調(diào)用時返回的數(shù) end #輸出1 2 3 4 5 6 7 8 9
當(dāng)然上例只是拿來做例子,實際上沒有人會這樣定義,更好的定義如下:
def nTime(n) (0..n).each {|v| yield(v)} end nTime(9) do |v| print "#{v} " end
我們來看下Array中的find實現(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 } #實現(xiàn)查找第一個大于5的數(shù),輸出7。
?因為塊的出現(xiàn),Ruby中少了許多for語句,代碼看上去更人性化,寫代碼不再是枯燥的事,而是一種享受。
四,傳遞塊的另一種方式
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中,提供了一個非常酷的特性,可以通過使用Objec#instance_eval(), Objec#instance_exec()方法插入一個代碼塊,做一個的對象上下文探針(Context Proble),深入到對象中的代碼片段,對其進行操作。有了這個特性以后,就可以很輕松的測試對象的行為,查看對象的當(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 上文是武林技術(shù)頻道小編和大家分享的詳解Ruby中的代碼塊及其參數(shù)傳遞,希望小編的分享對你受益匪淺,小編會在后面的日子里,繼續(xù)為大家搜集和整理這方面的知識。
新聞熱點
疑難解答
圖片精選