国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 語言 > JavaScript > 正文

JavaScript惰性求值的一種實現方法示例

2024-05-06 15:43:19
字體:
來源:轉載
供稿:網友

前言

在學習 Haskell 時,我遇到了這種寫法:

sum (takeWhile (<10000) (filter odd (map (^2) [1..])))

這段代碼的意思是,找出自然整數中小于 10000 的同時是乘方數和奇數的數字,再把這些數加總。由于 Haskell 的懶運算特性,上面的程序并不會立馬生成從 1 到 無限大的自然數列表,而是會等待 takeWhile 指令,再生成符合條件的列表。如果用 JS 來寫,很難寫出這么簡潔高表達性的代碼。一個可能的思路就是寫個 while 循環,然后找到符合條件的數進行加總。這個比較簡單,我就不演示了。

但是如果我們要用高階函數來模擬 Haskell 的寫法,就要想個辦法實現懶運算了。提到懶,首先想到的就是 Iterator 。沒人踢它一腳告訴它 next(),它會一直坐那兒不動的。

現在我們就來用 Iterator 來實現一個懶運算。

首先定義一個生成從 1 到無窮大自然數的 generator :

const numbers = function*() { let i = 1 while (true) { yield i++ }}

由于只有在 generator 執行后生成的 iterable 上執行 next() 方法,yield 才會執行,所以我們要做的主要工作就是實現不同的 next 方法,達到目的。

我們需要先創建一個工廠函數 Lazy,Lazy 封裝了我們的各種目標操作 :

const Lazy = iterator => { const next = iterable.next.bind(iterable) const map = () => {} const filter = () => {} const takeWhile = () => {} return {  next,  map,  filter,  takeWhile, }

我們先實現 map 方法,它會把每次 next 返回的值根據提供的回調函數進行修改:

const map = f => { const modifiedNext = () => { const item = next() const mappedValue = f(item.value) return {  value: mappedValue,  done: item.done, } } const newIter = { ...iterable, next: modifiedNext } return lazy(newIter)}

再定義 filter 方法,它會讓 next 只返回符合判斷條件的值:

const filter = predicate => { const modifiedNext = () => { while (true) {  const item = next()  if (predicate(item.value)) {  return item  } } } const newIter = { ...iterable, next: modifiedNext } return lazy(newIter)}

最后,定義 takeWhile,它會限制 next 執行的條件,一旦條件不滿足,則停止執行 next 并返回歷史執行結果:

const takeWhile = predicate => { const result = [] let value = next().value while (predicate(value)) { result.push(value) value = next().value } return result}

主要的方法都定義完了,現在把它們合并起來:

const Lazy = iterable => { const next = iterable.next.bind(iterable) const map = f => { const modifiedNext = () => {  const item = next()  const mappedValue = f(item.value)  return {  value: mappedValue,  done: item.done,  } } const newIter = { ...iterable, next: modifiedNext } return lazy(newIter) } const filter = predicate => { const modifiedNext = () => {  while (true) {  const item = next()  if (predicate(item.value)) {   return item  }  } } const newIter = { ...iterable, next: modifiedNext } return lazy(newIter) } const takeWhile = predicate => { const result = [] let value = next().value while (predicate(value)) {  result.push(value)  value = next().value } return result } return Object.freeze({ map, filter, takeWhile, next, })}const numbers = function*() { let i = 1 while (true) { yield i++ }}            
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

主站蜘蛛池模板: 松江区| 威宁| 福清市| 平邑县| 江阴市| 高青县| 双辽市| 张家界市| 田东县| 肇州县| 文山县| 寿阳县| 吉隆县| 嘉善县| 禹城市| 兰州市| 石嘴山市| 青铜峡市| 高清| 金坛市| 通州区| 张家口市| 延寿县| 金门县| 河源市| 萨迦县| 彰化县| 曲靖市| 涿州市| 乳山市| 姚安县| 邢台县| 竹溪县| 包头市| 漳浦县| 克东县| 通渭县| 翼城县| 陈巴尔虎旗| 湘潭市| 平湖市|