【譯】理解JavaScript中的柯里化

来源:https://www.cnblogs.com/GeniusLyzh/archive/2018/11/10/9937829.html
-Advertisement-
Play Games

譯文開始 函數式編程是一種編程風格,這種編程風格就是試圖將傳遞函數作為參數(即將作為回調函數)和返回一個函數,但沒有函數副作用(函數副作用即會改變程式的狀態)。 有很多語言採用這種編程風格,其中包括JavaScript、Haskell、Clojure、Erlang和Scala等一些很流行的編程語言。 ...


譯文開始

函數式編程是一種編程風格,這種編程風格就是試圖將傳遞函數作為參數(即將作為回調函數)和返回一個函數,但沒有函數副作用(函數副作用即會改變程式的狀態)。
有很多語言採用這種編程風格,其中包括JavaScript、Haskell、Clojure、Erlang和Scala等一些很流行的編程語言。
函數式編程憑藉其傳遞和返回函數的能力,帶來了許多概念:

  • 純函數
  • 柯里化
  • 高階函數
    其中一個我們將要看到的概念就是柯里化。
    在這篇文章,我們將看到柯里化是如何工作以及它如何在我們作為軟體開發者的工作中發揮作用。

什麼是柯里化

柯里化是函數式編程中的一種過程,可以將接受具有多個參數的函數轉化為一個的嵌套函數隊列,然後返回一個新的函數以及期望下一個的內聯參數。它不斷返回一個新函數(期望當前參數,就像我們之前說的那樣)直到所有參數都用完為止。這些參數會一直保持“存活”不會被銷毀(利用閉包的特性)以及當柯里化鏈中最後的函數返回並執行時,所有參數都用於執行。

柯里化就是將具有多個arity的函數轉化為具有較少的arity的函數。——kbrainwave
備註:術語arity(元數):指的是函數的參數個數,例如:

function fn(a, b) {
    //...
}
function _fn(a, b, c) {
    //...
}

函數fn有兩個參數(即 2-arity函數)以及_fn有三個參數(即3-arity函數)。
因此,柯里化將一個具有多個參數的函數轉化為一系列只需一個參數的函數。
下麵,我們看一個簡單的例子:

function multiply(a, b, c) {
    return a * b * c;
}

這個函數接收三個數字並且相乘,然後返回計算結果。

multiply(1,2,3); // 6

接下來看看,我們如何用完整參數調用乘法函數。我們來創建一個柯里化版本的函數,然後看看如何在一系列調用中調用相同的函數(並且得到同樣的結果)。

function multiply(a) {
    return (b) => {
        return (c) => {
            return a * b * c
        }
    }
}
log(multiply(1)(2)(3)) // 6

我們已經將multiply(1,2,3)函數調用形式轉化為multiply(1)(2)(3)多個函數調用的形式。
一個單獨的函數已經轉化為一系列的函數。為了得到三個數字1、2、3的乘法結果,這些數字一個接一個傳遞,每個數字會預先填充用作下一個函數內聯調用。
我們可以分開這個multiply(1)(2)(3)函數調用步驟,更好理解一點。

const mul1 = multiply(1);
const mul2 = mul1(2);
const result = mul2(3);
log(result); // 6

我們來一個接一個地傳遞參數。首先傳參數1到multiply函數:

let mul1 = multiply(1);

以上代碼執行會返回一個函數:

return (b) => {
        return (c) => {
            return a * b * c
        }
    }

現在,變數mul1會保持以上的函數定義,這個函數接收參數b。
我們調用函數mul1,傳入參數2:

let mul2 = mul1(2);

函數mul1執行後會返回第三個函數

return (c) => {
            return a * b * c
        }

這個返回的函數現在保存在變數mul2中。
本質上,變數mul2可以這麼理解:

mul2 = (c) => {
            return a * b * c
        }

當傳入參數3調用函數mul2時,

const result = mul2(3);

會使用之前傳入的參數進行計算:a=1,b=2,然後結果為6。

log(result); // 6

作為一個嵌套函數,mul2函數可以訪問外部函數的變數作用域,即multiply函數和mul1函數。
這就是為什麼mul2函數能使用已經執行完函數中定義的變數中進行乘法計算。雖然函數早已返回而且已經在記憶體中執行垃圾回收。但是它的變數還是以某種方式保持“存活”。

備註:以上變數保持存活是閉包特性,不明白可以查看閉包相關文章瞭解更多
你可以看到三個數字每次只傳遞一個參數應用於函數,並且每次都返回一個新的函數,值得所有的參數用完為止。
下麵來看一個其他的例子:

function volume(l,w,h) {
    return l * w * h;
}
const aCylinder = volume(100,20,90) // 180000

上面是一個計算任何實心形狀體積的函數。
這個柯里化版本將接受一個參數以及返回一個函數,該函數同樣也接受一個參數和返回一個新的函數。然後一直這樣迴圈/繼續,直到到達最後一個參數並返回最後一個函數。然後執行之前的參數和最後一個參數的乘法運算。

function volume(l) {
    return (w) => {
        return (h) => {
            return l * w * h
        }
    }
}
const aCylinder = volume(100)(20)(90) // 180000

就像之前的multiply函數那樣,最後的函數只接受一個參數h,但是仍然會對那些早已執行完返回的函數作用域中里的其他變數執行操作。能這樣操作是因為有閉包的特性。

譯者註:以上寫的很啰嗦,感覺另外的例子完全就是重覆說明。
柯里化背後的想法其實是獲取一個函數並派生出一個返回特殊函數的函數。

柯里化在數學方面的應用

我有點喜歡數學說明

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 鎖概述 MySQL的鎖機制,就是資料庫為了保證數據的一致性而設計的面對併發場景的一種規則。 最顯著的特點是不同的存儲引擎支持不同的鎖機制,InnoDB支持行鎖和表鎖,MyISAM支持表鎖。 表鎖就是把整張表鎖起來,特點是加鎖快,開銷小,不會出現死鎖,鎖粒度大,發生鎖衝突的概率高,併發相對較低。 行鎖 ...
  • mysql資料庫連接異常問題(總結) 1.1 前言 最近項目由1個數據源增加至了3個數據源(連接池使用C3P0),結果各種奇葩的資料庫連接問題接踵而至,為防止將來再次遇到同樣的問題不犯同樣錯誤,現總結如下。 1.2 An attempt by a client to checkout a Conne ...
  • 什麼是事務 事務是現代資料庫的核心概念之一,它表示資料庫一系列操作的集合。這些操作必須在一個事務當中,要麼全部執行成功,要麼全部不執行。 ACID原則 原子性(Atomicity) 原子性是指事務包含的所有操作要麼全部成功,要麼全部失敗回滾,因此事務的操作如果成功就必須要完全應用到資料庫,如果操作失 ...
  • 歡迎大家前往 "騰訊雲+社區" ,獲取更多騰訊海量技術實踐乾貨哦~ 本文由 "騰訊雲資料庫 TencentDB" 發表於 "雲+社區專欄" 今天我分享的主題內容大概是兩部分,最主要的還是小游戲和小程式,第一部分就是跟大家分享下我們在現網運營中服務小游戲以及爆款小游戲積累的經驗。在現網運維中我們做了一 ...
  • 作者:天山老妖S 鏈接:http://blog.51cto.com/9291927 一、事務簡介 1、事務簡介 事務(Transaction)是指作為單個邏輯工作單元執行的一系列操作。 2、事物的特效 A、原則性(Atomicity) 表示組成一個事物的多個資料庫操作是一個不可分隔的原子單元,只有所 ...
  • 分散式事務創建註意事項 指定一個由 Transact-SQL 分散式事務處理協調器 (MS DTC) 管理的 Microsoft 分散式事務的起點。 執行 BEGIN DISTRIBUTED TRANSACTION 語句的 SQL Server 資料庫引擎的實例是事務創建者,並控制事務的完成。 當為 ...
  • 歡迎大家前往 "騰訊雲+社區" ,獲取更多騰訊海量技術實踐乾貨哦~ 本文由 "goo" 發表於 "雲+社區專欄" 與生活緊密相連的音視頻,為何有那麼多格式?直播、點播以及即時視頻其中又有怎樣的機制支撐?面對紛繁複雜的音視頻知識,應該如何學起?快速探索,音視頻技術不再神秘。 前言 面對一門技術,我們熟 ...
  • Error:Execution failed for task ':app:transformDexArchiveWithExternalLibsDexMergerForDebug'.> java.lang.RuntimeException: java.lang.RuntimeException: ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...