a亚洲精品_精品国产91乱码一区二区三区_亚洲精品在线免费观看视频_欧美日韩亚洲国产综合_久久久久久久久久久成人_在线区

首頁 > 編程 > Ruby > 正文

探討Ruby中block的理解

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

  ruby.html" target="_blank">Ruby里的block一般翻譯成代碼塊,下面就一起跟小編來探討一下Ruby中block的理解吧,感興趣的朋友跟小編一起來了解一下吧!

  First-class function and Higher-order function

  First-class function 和 Higher-order function 是函數式編程語言里面的概念,聽起來好像很高端的樣子,其實很很簡單的。

  First-class functions 是指在某些語言里,函數是一等公民,可以把函數當做參數傳遞,

  可以返回一個函數,可以把函數賦值個一個變量等等,反正就是正常值能做的事函數都能做。JavaScript 就是這樣的。舉個例子(下面的所有例子里,當我提到

  JavaScript 時,示例代碼都用的 CoffeeScript):

  greet = (name) ->

  return -> console.log "Hello, #{name}"

  greetToMike = greet("Mike")

  greetToMike() # => 輸出 "Hello, Mike"

  a = greetToMike

  a() # => 輸出 "Hello, Mike"

  在上面的第四行里,greet("Mike") 返回了一個函數,所以第五行里才可以調用 greetToMike()輸出"Hello, Mike"。第六行把一個函數賦值給了a,所以第七行就可以調用這個函數了。

  higher-order function 一般翻譯成高階函數,是指接受函數做參數或者返回函數的函數。

  舉個非常常用的例子(用 JavaScript):

  a = [ "a", "b", "c", "d" ]

  a.map((x) -> x + '!') #=> ["a!", "b!", "c!", "d!"]

  上面例子里 map 就接受了一個匿名函數作為參數。Array.prototype里的很多方法,比如reduce, filter,every, some 等等都是高階函數,因為他們都接受函數作為參數。

  高階函數非常強大,表達力很強,可以避免大量重復代碼。總的來說,它就是個好東西。

  Block 的本質

  先來看一組 Ruby 和 CoffeeScript 代碼的對比。

  a = [ "a", "b", "c", "d" ]

  a.map { |x| x + "!" } # => ["a!", "b!", "c!", "d!"]

  a.reduce { |acc, x| acc + x} # => "abcd"

  a = [ "a", "b", "c", "d" ]

  a.map((x) -> x + '!') # => ["a!", "b!", "c!", "d!"]

  a.reduce((acc, x) -> acc + x) # => "abcd"

  這兩組代碼真的看起來超級像。我覺得這也暴露了 Ruby 的 block 的本質:高階函數的函數參數的變體。

  JavaScript 里面的map 函數接受一個函數作為參數,但是 Ruby 里的 map 卻接受一個

  block 作為參數。

  其實 matz 早在一本書里《松本行弘的程序世界》里說了:

  復制代碼 代碼如下:

  最終來看,塊到底是什么?

  ...

  塊也可以看作只是高階函數的一種特殊形式的語法。

  ...

  高階函數和塊的本質一樣

  ...

  在 Ruby 里,函數不是一等公民,沒有 first-class functions。但是在 Ruby

  里怎樣使用高階函數呢?答案就是使用 block。可以直接用 block,也可以用 lambda

  或者 proc 把 block 轉換成 Proc 類的實例用。

  我發現在 Ruby 里使用 block 時,幾乎所有的情況下都可以用 JavaScript

  的高階函數替代。

  Enumerable 模塊里的所有方法都是典型的例子。事實上確實存在 JavaScript 版

  的 Enumerable,比如 Prototype.js 就有個 Enumerable,用起來跟 Ruby版的幾乎一樣的。當然它是通過高階函數實現的。

  與高階函數有何不同

  除了語法上看上去有點不同外,有非常重要的兩點。

  控制流操作

  在 block 里面可以用 break, next 等等這些在一般的循環里才有的控制流操作,這些

  在高階函數里是用不了的。比如你可以試試在 JavaScript 里用 forEach 而不用循環

  實現個take_while 函數,真是相當別扭的。比如之前 cnode 上就有人發帖問:nodejs的forEach不支持break嗎?,其實這個帖子下面回復用 return 的基本上都是錯的,

  some 和 every 這樣利用 短路求值 的特點確實可以 hack 一下,但是明顯不自然而且大大增加了別人理解代碼的難度。

  從這一點來看 block 確實還不錯的。

  只有一個函數參數的高階函數

  Ruby 里一個方法只能接受一個 block 作為參數,大概就是類似于只有一個函數參數的高階

  函數。看起來好像是受到限制了。其實那本《松本行弘的程序世界》對此也有點解釋。

  大概是說了一個調查,在傾向于使用高階函數的 OCaml 的標準庫中,94%

  的高階函數只有一個函數參數。所以說這點限制不是什么問題。就我自己的體驗來說,在 JavaScript 里,還從沒用到需要兩個函數參數的高階函數。

  未說明的

  嗯,這篇文章看起來有點太長了,所以我不打算寫下去了。其實還有一些重要的地方沒說。比如

  Block 其實可以作為閉包用的。Ruby 里用def定義方法時有點悲劇的,因為它不是閉包,接觸

  不到它外面的變量。

  name = "mike"

  def greet

  puts "hello, #{name}"

  end

  hello # => in `greet': undefined local variable or method `name' for main:Object (NameError)

  但是用 block 就可以了

  name = "mike"

  define_method(:greet) do

  puts "hello, #{name}"

  end

  greet # => "hello, mike"

  用 JavaScript 就根本不存在問題。

  name = "mike"

  greet = -> console.log "hello, #{name}"

  greet() # => "hello, mike"

  同理還有class 和 module 關鍵字都會創建新的作用域而在里面接觸不到外面的變量,

  也可以用 block 解決。

  還有那個 proc 和 lambda 的區別。其實我一直不理解為什么會有人不用lambda

  而跑去用 proc,明顯 proc 的 return 行為太不符合常識了。但是到頭來卻發現

  block 的行為跟 proc 創建的對象的行為是一樣的,比如

  def hello

  (1..10).each { |e| return e}

  return "hello"

  end

  hello # => 1

  這感覺真是有點悲催。

  以上就是小編對于Ruby中block的理解了,更多相關內容請繼續關注武林技術頻道。

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

圖片精選

主站蜘蛛池模板: 久久精品免费视频观看 | 精品视频在线观看 | 91久久国产精品 | www.毛片 | 91精品国产高清一区二区三区 | 毛片av网站| 欧美精品免费在线观看 | 亚洲人成网亚洲欧洲无码 | 亚洲欧美日韩另类精品一区二区三区 | 日韩1| 超碰香蕉| 色婷婷综合久久久久中文一区二区 | 国产日韩精品一区二区在线观看播放 | 1000部羞羞视频在线看视频 | 国产精品中文字幕在线播放 | 国产一区免费在线 | 国产亚洲一区二区三区在线观看 | 日本不卡一区 | 欧美精品一区三区 | 亚洲精品久久久久久久久久久 | 精品国产一区二区三区久久久蜜月 | 午夜视| 婷婷桃色网 | 国产一级淫免费播放m | 黄色网址免费大全 | 久久精品免费电影 | 免费看男女www网站入口在线 | 婷婷久久综合 | 国产999精品久久久影片官网 | 天天色天天色 | 国产午夜精品一区二区三区视频 | 久久久网站| 久草视频在线播放 | 亚洲一区二区三区免费在线观看 | 秋霞一区 | 蜜桃久久av | 最新国产中文字幕 | 欧美亚洲午夜 | 免费av手机在线观看 | 在线欧美视频 | 欧美aⅴ一区二区 |