詳解Ruby中的代碼塊及其參數傳遞_ruby專題

來源:互聯網
上載者:User

一,塊的聲明 
  塊的聲明在函數調用之後,用{..}括起來,或do..end封裝。{}一般用在單行語句上,do..end用在多行語句上。

(1..4).each{|v| print "#{v} "} #輸出1 2 3 4 

  塊可以帶參數,與函數參數不同,塊參數用||封裝,當然,可以帶多個參數。這些參數怎麼定義,實際上是在函數內部定義好的,後面會講到。

二,塊內變數的訪問 
  塊內可以訪問塊外的變數,也就是塊外的變數在塊內是可見的,如

sum = 0 (1..5).each do |v|   name = 'smile' #name屬於塊內變數,其可視範圍只能在塊內。假設塊外沒有相同名稱的變數.   sum += v #sum在塊內可見 end p sum #輸出15,sum已改變。 p name #Error! name不可訪問。 

  正因塊內可以塊外的變數所以可能不小心修改了一些外部變數,這是我們不希望的。幸運的是Ruby1.9版本後,提供了一種安全的方式聲明塊內變數,在塊參數後面加";",塊內變數放在";"之後.

name = 'outside' sum = 0 (1..5).each do |v;name| #name在";"之後,可以聲明多個變數,用逗號隔開   name = 'inside' #name屬於塊內變數,其可視範圍只能在塊內.假設塊外沒有相同名稱的變數。   sum += v #sum在塊內可訪問 end p sum #輸出15,sum已改變。 p name #輸出outside,沒有變。 

三,yield語句 
  看這裡,可能還不是很明白,函數是如何調用塊的。現在就來介紹塊的調用,關鍵是yield語句。在函數體中,如果用yield,函數會調用函數的塊。

def threeTime   yield   yield   yield end threeTime{p 'Hello world!'} 

  輸出三行Hello world!,是不是很簡單呢。現在應該明白了吧,是yield調用的塊。
塊的參數是怎麼回事呢?估計你已經想到了,就是yield的參數,跟一般函數一樣yield可以帶參數的。看例子

def takeBlock(p1)  if block_given? # 判斷是否有塊,如果在yield時,沒有聲明塊,會出錯,所以在這裡作判斷會好點。   yield(p1) #把p1傳給塊參數,既下面塊聲明中的s  else   p1  end endie  takeBlock("no block")  #輸出"no block" takeBlock("no block") { |s| s.sub(/no /, '') } #輸出"block" 

  既然yield能傳參數給塊,反過來,塊能不能傳值給yield呢?答案是肯定的。塊中最後一句語句的值會自動傳給yield。請看樣本

def nTime  i = yield #第一次調用時,返回塊的值  (0..i).each {|v| yield(v)} # 此處yield也可以放在塊中 end nTime do |v|  print "#{v} " if v  9 #yield調用時返回的數 end #輸出1 2 3 4 5 6 7 8 9 

當然上例只是拿來做例子,實際上沒有人會這樣定義,更好的定義如下:

def nTime(n)  (0..n).each {|v| yield(v)} end nTime(9) do |v|  print "#{v} " end 

我們來看下Array中的find實現

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 } #實現尋找第一個大於5的數,輸出7。

 因為塊的出現,Ruby中少了許多for語句,代碼看上去更人性化,寫代碼不再是枯燥的事,而是一種享受。

四,傳遞塊的另一種方式

def fun #不帶參數的  yield end proc = ->{p 'haha'}  fun &proc ##### def fun2(x) #帶參數的  yield x end proc2 = ->(x){p x} fun2 1,&proc2 

五,instance_eval()和instance_exec()
在Ruby中,提供了一個非常酷的特性,可以通過使用Objec#instance_eval(), Objec#instance_exec()方法插入一個代碼塊,做一個的物件內容探針(Context Proble),深入到對象中的程式碼片段,對其進行操作。有了這個特性以後,就可以很輕鬆的測試對象的行為,查看對象的目前狀態。

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 

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.