真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

JS中柯里化currying的示例分析

這篇文章主要為大家展示了“JS中柯里化currying的示例分析”,內(nèi)容簡而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領(lǐng)大家一起研究并學(xué)習(xí)一下“JS中柯里化currying的示例分析”這篇文章吧。

我們提供的服務(wù)有:成都網(wǎng)站建設(shè)、成都網(wǎng)站設(shè)計、微信公眾號開發(fā)、網(wǎng)站優(yōu)化、網(wǎng)站認(rèn)證、鳳城ssl等。為成百上千家企事業(yè)單位解決了網(wǎng)站和推廣的問題。提供周到的售前咨詢和貼心的售后服務(wù),是有科學(xué)管理、有技術(shù)的鳳城網(wǎng)站制作公司

何為Curry化/柯里化?

curry化來源與數(shù)學(xué)家 Haskell Curry的名字 (編程語言 Haskell也是以他的名字命名)。

柯里化通常也稱部分求值,其含義是給函數(shù)分步傳遞參數(shù),每次傳遞參數(shù)后部分應(yīng)用參數(shù),并返回一個更具體的函數(shù)接受剩下的參數(shù),這中間可嵌套多層這樣的接受部分參數(shù)函數(shù),直至返回最后結(jié)果。

因此柯里化的過程是逐步傳參,逐步縮小函數(shù)的適用范圍,逐步求解的過程。

柯里化一個求和函數(shù)

按照分步求值,我們看一個簡單的例子

var concat3Words = function (a, b, c) {
 return a+b+c;
};
var concat3WordsCurrying = function(a) {
 return function (b) {
  return function (c) {
   return a+b+c;
  };
 };
};
console.log(concat3Words("foo ","bar ","baza"));   // foo bar baza
console.log(concat3WordsCurrying("foo "));     // [Function]
console.log(concat3WordsCurrying("foo ")("bar ")("baza")); // foo bar baza

可以看到, concat3WordsCurrying("foo ") 是一個 Function,每次調(diào)用都返回一個新的函數(shù),該函數(shù)接受另一個調(diào)用,然后又返回一個新的函數(shù),直至最后返回結(jié)果,分布求解,層層遞進。(PS:這里利用了閉包的特點)

那么現(xiàn)在我們更進一步,如果要求可傳遞的參數(shù)不止3個,可以傳任意多個參數(shù),當(dāng)不傳參數(shù)時輸出結(jié)果?

首先來個普通的實現(xiàn):

var add = function(items){
 return items.reduce(function(a,b){
  return a+b
 });
};
console.log(add([1,2,3,4]));

但如果要求把每個數(shù)乘以10之后再相加,那么:

var add = function (items,multi) {
 return items.map(function (item) {
  return item*multi;
 }).reduce(function (a, b) {
  return a + b
 });
};
console.log(add([1, 2, 3, 4],10));

好在有 map 和 reduce 函數(shù),假如按照這個模式,現(xiàn)在要把每項加1,再匯總,那么我們需要更換map中的函數(shù)。

下面看一下柯里化實現(xiàn):

var adder = function () {
 var _args = [];
 return function () {
  if (arguments.length === 0) {
   return _args.reduce(function (a, b) {
    return a + b;
   });
  }
  [].push.apply(_args, [].slice.call(arguments));
  return arguments.callee;
 }
}; 
var sum = adder();
console.log(sum);  // Function
sum(100,200)(300); // 調(diào)用形式靈活,一次調(diào)用可輸入一個或者多個參數(shù),并且支持鏈?zhǔn)秸{(diào)用
sum(400);
console.log(sum()); // 1000 (加總計算)

上面 adder是柯里化了的函數(shù),它返回一個新的函數(shù),新的函數(shù)接收可分批次接受新的參數(shù),延遲到最后一次計算。

通用的柯里化函數(shù)

更典型的柯里化會把最后一次的計算封裝進一個函數(shù)中,再把這個函數(shù)作為參數(shù)傳入柯里化函數(shù),這樣即清晰,又靈活。

例如 每項乘以10, 我們可以把處理函數(shù)作為參數(shù)傳入:

var currying = function (fn) {
 var _args = [];
 return function () {
  if (arguments.length === 0) {
   return fn.apply(this, _args);
  }
  Array.prototype.push.apply(_args, [].slice.call(arguments));
  return arguments.callee;
 }
};
var multi=function () {
 var total = 0;
 for (var i = 0, c; c = arguments[i++];) {
  total += c;
 }
 return total;
};
var sum = currying(multi); 
sum(100,200)(300);
sum(400);
console.log(sum());  // 1000 (空白調(diào)用時才真正計算)

這樣 sum = currying(multi),調(diào)用非常清晰,使用效果也堪稱絢麗,例如要累加多個值,可以把多個值作為做個參數(shù) sum(1,2,3),也可以支持鏈?zhǔn)降恼{(diào)用,sum(1)(2)(3)

柯里化的作用

  • 延遲計算。上面的例子已經(jīng)比較好低說明了。

  • 參數(shù)復(fù)用。當(dāng)在多次調(diào)用同一個函數(shù),并且傳遞的參數(shù)絕大多數(shù)是相同的,那么該函數(shù)可能是一個很好的柯里化候選。

  • 動態(tài)創(chuàng)建函數(shù)。

這可以是在部分計算出結(jié)果后,在此基礎(chǔ)上動態(tài)生成新的函數(shù)處理后面的業(yè)務(wù),這樣省略了重復(fù)計算?;蛘呖梢酝ㄟ^將要傳入調(diào)用函數(shù)的參數(shù)子集,部分應(yīng)用到函數(shù)中,從而動態(tài)創(chuàng)造出一個新函數(shù),這個新函數(shù)保存了重復(fù)傳入的參數(shù)(以后不必每次都傳)。例如,事件瀏覽器添加事件的輔助方法:

var addEvent = function(el, type, fn, capture) {
  if (window.addEventListener) {
   el.addEventListener(type, function(e) {
    fn.call(el, e);
   }, capture);
  } else if (window.attachEvent) {
   el.attachEvent("on" + type, function(e) {
    fn.call(el, e);
   });
  } 
 };

每次添加事件處理都要執(zhí)行一遍 if...else...,其實在一個瀏覽器中只要一次判定就可以了,把根據(jù)一次判定之后的結(jié)果動態(tài)生成新的函數(shù),以后就不必重新計算。

var addEvent = (function(){
 if (window.addEventListener) {
  return function(el, sType, fn, capture) {
   el.addEventListener(sType, function(e) {
    fn.call(el, e);
   }, (capture));
  };
 } else if (window.attachEvent) {
  return function(el, sType, fn, capture) {
   el.attachEvent("on" + sType, function(e) {
    fn.call(el, e);
   });
  };
 }
})();

這個例子,第一次 if...else... 判斷之后,完成了部分計算,動態(tài)創(chuàng)建新的函數(shù)來處理后面?zhèn)魅氲膮?shù),這是一個典型的柯里化。

Function.prototype.bind 方法也是柯里化應(yīng)用

與 call/apply 方法直接執(zhí)行不同,bind 方法 將第一個參數(shù)設(shè)置為函數(shù)執(zhí)行的上下文,其他參數(shù)依次傳遞給調(diào)用方法(函數(shù)的主體本身不執(zhí)行,可以看成是延遲執(zhí)行),并動態(tài)創(chuàng)建返回一個新的函數(shù), 這符合柯里化特點。

var foo = {x: 888};
var bar = function () {
 console.log(this.x);
}.bind(foo);    // 綁定
bar(); 
// 888

與 call/apply 方法直接執(zhí)行不同,bind 方法 將第一個參數(shù)設(shè)置為函數(shù)執(zhí)行的上下文,其他參數(shù)依次傳遞給調(diào)用方法(函數(shù)的主體本身不執(zhí)行,可以看成是延遲執(zhí)行),并動態(tài)創(chuàng)建返回一個新的函數(shù), 這符合柯里化特點。

var foo = {x: 888};
var bar = function () {
 console.log(this.x);
}.bind(foo);    // 綁定
bar(); 
// 888

下面是一個 bind 函數(shù)的模擬,testBind 創(chuàng)建并返回新的函數(shù),在新的函數(shù)中將真正要執(zhí)行業(yè)務(wù)的函數(shù)綁定到實參傳入的上下文,延遲執(zhí)行了。

Function.prototype.testBind = function (scope) {
 var fn = this;     //// this 指向的是調(diào)用 testBind 方法的一個函數(shù), 
 return function () {
  return fn.apply(scope);
 }
};
var testBindBar = bar.testBind(foo); // 綁定 foo,延遲執(zhí)行
console.log(testBindBar);    // Function (可見,bind之后返回的是一個延遲執(zhí)行的新函數(shù))
testBindBar();

這里要注意 prototype 中 this 的理解。

實例

實例1:

var currying = function(fn) {
 // fn 指官員消化老婆的手段
 var args = [].slice.call(arguments, 1);
 // args 指的是那個合法老婆
 return function() {
  // 已經(jīng)有的老婆和新搞定的老婆們合成一體,方便控制
  var newArgs = args.concat([].slice.call(arguments));
  // 這些老婆們用 fn 這個手段消化利用,完成韋小寶前輩的壯舉并返回
  return fn.apply(null, newArgs);
 };
};
// 下為官員如何搞定7個老婆的測試
// 獲得合法老婆
var getWife = currying(function() {
 var allWife = [].slice.call(arguments);
 // allwife 就是所有的老婆的,包括暗渡陳倉進來的老婆
 console.log(allWife.join(";"));
}, "合法老婆");
// 獲得其他6個老婆
getWife("大老婆","小老婆","俏老婆","刁蠻老婆","乖老婆","送上門老婆");
// 換一批老婆
getWife("超越韋小寶的老婆");
 結(jié)果:
合法老婆;大老婆;小老婆;俏老婆;刁蠻老婆;乖老婆;送上門老婆
合法老婆;超越韋小寶的老婆
實例2:
var curryWeight = function(fn) {
 var _fishWeight = [];
 return function() {
  if (arguments.length === 0) {
   return fn.apply(null, _fishWeight);
  } else {
   _fishWeight = _fishWeight.concat([].slice.call(arguments));
  }
 }
};
var fishWeight = 0;
var addWeight = curryWeight(function() {
 var i=0; len = arguments.length;
 for (i; i

以上是“JS中柯里化currying的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道!


網(wǎng)站題目:JS中柯里化currying的示例分析
文章鏈接:http://weahome.cn/article/iipogo.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部