麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁 > 編程 > Ruby > 正文

總結Ruby中的block、proc、lambda區別

2020-02-24 15:39:50
字體:
來源:轉載
供稿:網友

Ruby的閉包使用是非常頻繁的,為了徹底理解代碼,小編再次仔細研究了Ruby的閉包,特別是block、proc和lambda幾種用法的異同,今天這篇文章是武林技術頻道小編和你分享的經驗之談。

閉包是 Ruby 相對其它語言特別優勢之一,很多語言有閉包,但是唯有 Ruby 把閉包的優勢發揮得淋漓盡致。Ruby 的哲學之一:同一個問題現實中有多種解決方法,所以 Ruby 中也有多種解法,所以閉包的使用也有多種方法。

先看一個代碼的例子:

Example 1:

復制代碼 代碼如下:

def foo1
? yield
end

?

def foo2(&b)
? b.call if b
end

foo1 { puts "foo1 in block" }
foo2 { puts "foo2 in block" }
proc = Proc.new { puts "foo in proc" }
foo1(&proc)
foo2(&proc)
lambda_proc = lambda { puts "foo in lambda" }
foo1(&lambda_proc)
foo2(&lambda_proc)

?

輸出:

復制代碼 代碼如下:

》foo1 in block
》foo2 in block
》foo in proc
》foo in proc
》foo in lambda
》foo in lambda

?

大家是不是有些困惑,首先是方法 foo1 和 foo2 都能接收閉包,它們這兩種寫法有什么區別,然后是作為參數的三種閉包 block,proc和 lambda有什么區別。

1. yield 和 block call 的區別

yield 和 block call 兩種都能實現運行閉包,從實際運行效果來說,區別不大。其區別主要在于:

1.1 閉包的傳遞和保存

因為閉包已經傳遞到參數里,所以可以繼續傳遞或保存起來,例如:

Exampl 2:

復制代碼 代碼如下:

?class A
????? def foo1(&b)
???????? @proc = b
????? end
????? def foo2
????????? @proc.call if @proc
????? end
?? end

?

??? a = A.new
??? a.foo1 { puts "proc from foo1" }
??? a.foo2

?

1.2 性能

yield不是方法調用,而是 Ruby 的關鍵字,yield 要比 block call 比快 1 倍左右。

2. block 和 proc, lambda 的區別

很簡單直接,引入 proc 和 lambda 就是為了復用,避免重復定義,例如在 example 1 中,使用 proc 變量存儲閉包,避免重復定義兩個 block 。

3. proc 和 lambda 的區別

這大概是最讓人困惑的地方,從 Example 1 的行為上看,他們的效果是一致的,為什么要用兩種不同的表達方式。

復制代碼 代碼如下:

?proc = Proc.new { puts "foo in proc" }
?? lambda_proc = lambda { puts "foo in lambda" }

?

確實,對于簡單的情況,比如 Example 1的情況,他們的行為是一致的,但是主要在兩個地方有明顯差異:

1.1 參數檢查

還是例子說話 Example 3:

復制代碼 代碼如下:

def foo
????? x = 100
????? yield x
?? end

?

?? proc = Proc.new { |a, b| puts "a is #{a.inspect} b is #{b.inspect}" }
?? foo(&proc)


?? lambda_proc1 = lambda { |a| puts "a is #{a.inspect}" }
?? foo(&lambda_proc1)
?? lambda_proc2 = lambda { |a, b| puts "a is #{a.inspect} b is #{b.inspect}" }
?? foo(&lambda_proc2)

?

輸出

?

復制代碼 代碼如下:

?? 》a is 100 b is nil
?? 》a is 100
?? 》ArgumentError: wrong number of arguments (1 for 2)
????? …

?

可見,proc 不會對參數進行個數匹配檢查,而 lambda 會,如果不匹配還會報異常,所以安全起見,建議優先用 lambda。

1.2 返回上層

還是例子說話 Example 4:

?

復制代碼 代碼如下:

?def foo
???? f = Proc.new { return "return from foo from inside proc" }
???? f.call # control leaves foo here
???? return "return from foo"
?? end

?


?? def bar
???? f = lambda { return "return from lambda" }
???? puts f.call # control does not leave bar here
???? return "return from bar"
?? end


?? puts foo
?? puts bar

?

輸出

?

復制代碼 代碼如下:

?? 》return from foo from inside proc
?? 》return from lambda
?? 》return from bar

?

可見,proc 中,return 相當于執行了閉包環境里的 return,而 lambda 只是返回call 閉包所在環境。

閉包是 Ruby 的強大特性,它的幾種表達方式block,proc 和 lambda有細微差別,要用好它需要對其深入理解。

閉包是Ruby的一個強大特性,而block、proc和lambda的幾個表達式有細微的差別。為了更好地使用它,武林技術頻道小編帶大家來學習吧!

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

主站蜘蛛池模板: 视频一区国产精品 | 国产亚洲精品成人a | 国产精品视频一区二区三区四区国 | 麻豆视频网 | 日韩黄网站 | 天天看逼 | 色妞欧美 | 国产一级一区二区三区 | 逼特逼视频在线观看 | 在线成人av观看 | 国产午夜免费福利 | 国产一国产精品一级毛片 | 一级免费视频 | 欧美一级性 | 欧美视频在线一区二区三区 | 日本大片在线播放 | 亚洲第一成人在线视频 | 男女一边摸一边做羞羞视频免费 | 国产一国产精品一级毛片 | 最新在线中文字幕 | 91系列在线观看 | 欧美成人国产va精品日本一级 | 中文字幕涩涩久久乱小说 | 夜夜看| 免费一级毛片在线播放视频 | 国产精品成人av片免费看最爱 | 片在线观看 | 国产精品视频一区二区三区四区五区 | 久久99精品久久久久久国产越南 | 久久99精品久久久久久秒播放器 | 懂色av懂色aⅴ精彩av | 色女人在线 | 欧美xxxwww| 国产午夜精品久久久久 | 日本欧美在线播放 | 国产精品午夜在线观看 | 欧美成人一级 | 国产精品一区二区三区在线播放 | 久草手机在线 | 777午夜精品视频在线播放 | 91av网址|