成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

談談柯里化方式的累加器實現(xiàn)

魏憲會 / 2394人閱讀

摘要:最直接的方式當然是遍歷數組并累加得到結果,也可以使用數組的方法實現(xiàn),如下結合第四步,替換中的返回值即可將其進行簡化,得到最終結果當然,采用這種實現(xiàn)方式,對于形如的調用方式也是沒有問題的。

談談 JavaScript 中形如 add(1)(2)(3)(4) = 10 這種累加器方法的實現(xiàn)過程和思路

第一步:實現(xiàn)級聯(lián)

若是想要實現(xiàn) fn()() 這種調用方式的函數,則在 fn 函數體內一定也會返回一個函數,如下:

function fn(){
    return function(){
        
    }
}

再進一步,若是想要實現(xiàn) fn()()()... 不定次數的調用,則意味著每一層的返回值均為一個函數,這就需要使用類似遞歸的方式去實現(xiàn):

function fn(){
    var _fn = function(){
        return _fn;
    }
    return _fn;
}
第二步:實現(xiàn)返回值

如果每一層的返回值均為函數,那么如何返回累加結果呢?也就是說,在函數調用的最后一層,這個返回值應該是一個值而非函數。這里則需要使用 valueOf 這一方法,例子如下:

function fn1(){

}
console.log(fn1); // function ...

function fn2(){
    
}
fn2.valueOf = function(){
    return 1;
}
console.log(fn2); // function 1

注意,fn2 在控制臺的輸出結果為 function 1,但其數值型隱式轉換的結果為 1,也就是說 console.log(+fn2) 或是 var num = fn2 + 1 是可以正常地作為數值型進行計算的。而 fn2() 的方式仍然可以將其作為函數運算。

因而,我們只要給 _fn 添加 valueOf 方法就可以實現(xiàn)在不定次調用之后,得到一個「值」而非「函數」,如下:

function fn(){
    var _fn = function(){
        return _fn;
    }
    _fn.valueOf = function(){
        return 12345;
    }
    return _fn;
}

這里也可以使用添加 toString() 的方式實現(xiàn)這一功能,與 valueOf() 做法一致。

第三步:獲得傳入的參數

這一步相對簡單,使用內置對象 arguments 即可實現(xiàn)對函數中所傳入的參數的獲取,如下:

function foo(){
    console.log(arguments);
}

foo(1,2,3);

注意,這里的 arguments 不是數組,而是對象:

{
    0 : 1,
    1 : 2,
    2 : 3,
    callee : function foo(),
    length : 3,
    Symbol(Symbol.iterator) : function values(),
    __proto__ : Object,
}

若要將其轉換為數組,可以使用如下方式:

var arr = [].slice.call(arguments);
// 或是
var arr = Array.prototype.slice.call(arguments);

若一個對象含有 length 屬性,則可以通過這種方式轉換為數組形式

第四步:實現(xiàn)參數保存

add(1)(2)(3)... 累加器是在最后一次調用后返回之前所有參數的累加和。也就是說我們需要有一個地方可以保存先前的值或是計算結果。

在之前的代碼框架下,顯然不能將其保存在內層的 _fn 中。因為每層調用都相當于又一次的 _fn() 執(zhí)行,在其中定義的變量會被覆蓋。

使用全局變量當然是一種方式,但是這樣會污染全局空間,不是最佳方案。

考慮到對 fn()()()... 的調用實際返回的是內層的 _fn,意味著 fn 的局部變量其實也相當于 _fn 的全局變量。因而可以將保存先前參數的職責交給 fn 中的一個變量,代碼如下:

function fn(){
    var numList = [];
    var _fn = function(){
    
        // 這里測試思路是否可行
        numList.push(1);
        console.log(numList);
        
        return _fn;
    }
    _fn.valueOf = function(){
        return 12345;
    }
    return _fn;
}

console.log(fn()()()); // [1, 1]
// 注意這里雖然調用三次,但實際只執(zhí)行了兩次 push(1),第一次調用沒有執(zhí)行內層的 _fn(),而只是返回了它。

結合第三步,我們可以通過 push() 或是 concat() 的方式將每一次的參數組合起來,如下:

function fn(){
    var numList = [].slice.call(arguments);
    var _fn = function(){
        // 注意這里的 arguments 是傳入 _fn 的參數
        var innerArguments = [].slice.call(arguments);
        numList = numList.concat(innerArguments);
        console.log(numList);
        
        return _fn;
    }
    _fn.valueOf = function(){
        return 12345;
    }
    return _fn;
}

console.log(fn(1)(2)(3)); // [1, 2, 3]

當然,這里也可以使用 push() 的方式,將每一次的參數推入數組。

這一步還有另一種思路:用每一次的求和代替參數數組的合并。

第五步:求和計算

既然已經得到了全部的參數集合,對其進行求和就比較簡單了。最直接的方式當然是遍歷數組并累加得到結果,也可以使用數組的 reduce 方法實現(xiàn),如下:

var arr = [1, 2, 3];
var sum = arr.reduce(function(num1, num2){
    return num1 + num2;
});
console.log(sum); // 6

結合第四步,替換 valueOf 中的返回值即可:

function fn(){
    var numList = [].slice.call(arguments);
    var _fn = function(){
        var innerArguments = [].slice.call(arguments);
        numList = numList.concat(innerArguments);
        
        return _fn;
    }
    _fn.valueOf = function(){
        return numList.reduce(function(num1, num2){
            return num1 + num2;
        });
    }
    return _fn;
}

console.log(fn(1)(2)(3));

將其進行簡化,得到最終結果:

function fn(){
    var numList = [].slice.call(arguments);
    var _fn = function(){
        numList = numList.concat([].slice.call(arguments));
        return _fn;
    }
    _fn.valueOf = function(){
        return numList.reduce(function(i, j){return i+j;});
    }
    return _fn;
}

當然,采用這種實現(xiàn)方式,對于形如 fn(1, 2, 3)(4)(5, 6, 7) 的調用方式也是沒有問題的。

參考

前端基礎進階(八):深入詳解函數的柯里化 - 簡書

JS中的call()和apply()方法 - ITeye

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉載請注明本文地址:http://systransis.cn/yun/81858.html

相關文章

  • 從一道面試題談談函數柯里(Currying)

    摘要:忍者秘籍一書中,對于柯里化的定義如下在一個函數中首先填充幾個參數然后再返回一個新函數的技術稱為柯里化?;氐轿覀兊念}目本身,其實根據測試用例我們可以發(fā)現(xiàn),函數的要求就是接受單一函數,例如但是與柯里化不同之處在于,柯里化返回的一個新函數。   歡迎大家再一次來到我的文章專欄:從面試題中我們能學到什么,各位同行小伙伴是否已經開始了悠閑的春節(jié)假期呢?在這里提前祝大家雞年大吉吧~哈哈,之前有人說...

    cppprimer 評論0 收藏0
  • 「前端面試題系列6」理解函數柯里

    摘要:原題如下寫一個方法,當使用下面的語法調用時,能正常工作這道題要考察的,就是對函數柯里化的理解。當參數只有一個的時候,進行柯里化的處理。這其實就是函數柯里化的簡單應用。 showImg(https://segmentfault.com/img/bVbopGm?w=620&h=350); 前言 這是前端面試題系列的第 6 篇,你可能錯過了前面的篇章,可以在這里找到: ES6 中箭頭函數的...

    liaorio 評論0 收藏0
  • js 擴展 -- currying 柯里函數

    摘要:里也有柯里化的實現(xiàn),只是平時沒有在意。如果函數柯里化后雖然生搬硬套,不過現(xiàn)實業(yè)務也會有類似場景。 柯里化 先解釋下什么是 柯里化 在計算機科學中,柯里化(英語:Currying),又譯為卡瑞化或加里化,是把接受多個參數的函數變換成接受一個單一參數(最初函數的第一個參數)的函數,并且返回接受余下的參數而且返回結果的新函數的技術。 js 里也有柯里化的實現(xiàn),只是平時沒有在意。先把原文簡介貼...

    Pocher 評論0 收藏0
  • JavaScript函數式編程入門經典

    摘要:函數式編程的定義函數是一段可以通過其名稱被調用的代碼。純函數大多數函數式編程的好處來自于編寫純函數,純函數是對給定的輸入返回相同的輸出的函數,并且純函數不應依賴任何外部變量,也不應改變任何外部變量。 一個持續(xù)更新的github筆記,鏈接地址:Front-End-Basics,可以watch,也可以star。 此篇文章的地址:JavaScript函數式編程入門經典 正文開始 什么是函...

    silvertheo 評論0 收藏0
  • 邂逅函數柯里

    摘要:柯里化函數的作用函數柯里化允許和鼓勵你分隔復雜功能變成更小更容易分析的部分。指向的是調用方法的一個函數,綁定,延遲執(zhí)行可見,之后返回的是一個延遲執(zhí)行的新函數關于性能的備注通常,使用柯里化會有一些開銷。 引子 有這樣一道題目,實現(xiàn)一個函數,實現(xiàn)如下功能: var result = sum(1)(2)(3); console.log(result);//6 這道題目,印象中是一道技術筆試...

    Kross 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<