以下內容通過代碼講解和實例分析了JS中精巧的自動柯里化實現方法,并分析了柯里化函數的基礎用法和知識,學習一下吧。
什么是柯里化?
在計算機科學中,柯里化(Currying)是把接受多個參數的函數變換成接受一個單一參數(最初函數的第一個參數)的函數,并且返回接受余下的參數且返回結果的新函數的技術。這個技術由 Christopher Strachey 以邏輯學家 Haskell Curry 命名的,盡管它是 Moses Schnfinkel 和 Gottlob Frege 發明的。
理論看著頭大?沒關系,先看看代碼:
柯里化應用
假設我們需要實現一個對列表元素進行某種處理的功能,比如說讓列表內每一個元素加一,那么很容易想到:
const list = [0, 1, 2, 3];list.map(elem => elem + 1);
很簡單是吧?如果又要加2呢?
const list = [0, 1, 2, 3];list.map(elem => elem + 1);list.map(elem => elem + 2);
看上去效率有點低,處理函數封裝下?
可是map的回調函數只接受當前元素 elem 這一個參數,看上去好像沒有辦法封裝...
你也許會想:如果能拿到一個部分配置好的函數就好了,比如說:
// plus返回部分配置好的函數const plus1 = plus(1);const plus2 = plus(2);plus1(5); // => 6plus2(7); // => 9
把這樣的函數傳進map:
const list = [0, 1, 2, 3];list.map(plus1); // => [1, 2, 3, 4]list.map(plus2); // => [2, 3, 4, 5]
是不是很棒棒?這樣一來不管是加多少,只需要list.map(plus(x))就好了,完美實現了封裝,可讀性大大提高!
不過問題來了:這樣的plus函數要怎么實現呢?
這時候柯里化就能派上用場了:
柯里化函數
// 原始的加法函數function origPlus(a, b) { return a + b;}// 柯里化后的plus函數function plus(a) { return function(b) { return a + b; }}// ES6寫法const plus = a => b => a + b;可以看到,柯里化的 plus 函數首先接受一個參數 a,然后返回一個接受一個參數 b 的函數,由于閉包的原因,返回的函數可以訪問到父函數的參數 a,所以舉個例子:const plus2 = plus(2)就可等效視為function plus2(b) { return 2 + b; },這樣就實現了部分配置。
通俗地講,柯里化就是一個部分配置多參數函數的過程,每一步都返回一個接受單個參數的部分配置好的函數。一些極端的情況可能需要分很多次來部分配置一個函數,比如說多次相加:
multiPlus(1)(2)(3); // => 6
這種寫法看著很奇怪吧?不過如果入了JS的函數式編程這個大坑的話,這會是常態。
JS中自動柯里化的精巧實現
新聞熱點
疑難解答
圖片精選