前端开发者进阶之函数反柯里化unCurrying

函数柯里化,是固定部分参数,返回一个接受剩余参数的函数,也称为部分计算函数,目的是为了缩小适用范围,建立一个针对性更强的函数。javascript

那么反柯里化函数,从字面讲,意义和用法跟函数柯里化相比正好相反,扩大适用范围,建立一个应用范围更广的函数。使原本只有特定对象才适用的方法,扩展到更多的对象。java

看一下通用函数:数组

Function.prototype.currying = function() {
    var that = this;
    return function() {
        return Function.prototype.call.apply(that, arguments);
    }
}

短小精悍,科学上讲,浓缩的都是精品,但越精品的每每越难以理解。分解一下:app

1 为Function原型添加unCurrying方法,这样全部的function均可以被借用;函数

2 返回一个借用其它方法的函数,这是目的;this

3 借用call方法实现,但call方法参数传入呢?借用apply,至此完毕。spa

回头看看,好像也不难!prototype

还有其它的实现方式:设计

Function.prototype.unCurrying = function () {
    var f = this;
    return function () {
        var a = arguments;
        return f.apply(a[0], [].slice.call(a, 1));
    };
};
Function.prototype.unCurrying = function () {
    return this.call.bind(this);
};

 原理都相同,最终是把this.method转化成 method(this,arg1,arg2....)以实现方法借用和this的泛化。code

鸭式辩型:若是一个对象能够像鸭子同样走路,游泳,而且嘎嘎叫,就认为这个对象是鸭子,哪怕它并非从鸭子对象继承过来的。

在javascript里面,不少函数都不作对象的类型检测,而是只关心这些对象能作什么。

Array构造器和String构造器的prototype上的方法就被特地设计成了鸭子类型。这些方法不对this的数据类型作任何校验。这也就是为何arguments能冒充array调用push方法.
看下v8引擎里面Array.prototype.push的代码:
function ArrayPush() {
    var n = TO_UINT32(this.length);
    var m = %_ArgumentsLength();
    for (var i = 0; i < m; i++) {
        this[i + n] = %_Arguments(i); //属性拷贝
        this.length = n + m; //修正length
        return this.length;
    }
}

这就给对象冒充创造了条件,也就咱们讨论的函数柯反里化unCurrying。反柯里化其实反映的是一种思想,扩大方法的适用范围!

看下面例了,让一个普通对象具有push方法:

var push = Array.prototype.push.unCurrying(),
obj = {};
push(obj, 'first', 'two');
console.log(obj);
/*obj {
    0 : "first",
    1 : "two"
}*/

obj被push了两个元素:first 和two,同时还具有了一个length属性,其实咱们建立了一类数组对象。

 再看一个例子:

var toUpperCase = String.prototype.toUpperCase.unCurrying();
console.log(toUpperCase('avd')); // AVD

function AryUpper(ary) {
    return ary.map(toUpperCase);
}

console.log(AryUpper(['a', 'b', 'c'])); // ["A", "B", "C"]

只是方法均可以借用。包括call方法。

var call = Function.prototype.call.unCurrying();
function $(id) {
    return this.getElementById(id);
}
var demo = call($, document, 'demo');
console.log(demo);

这彷佛看起来相对于前两个例子,比较难理解,其实一句话就能够解释:document借用了$方法,并替换了其中的this。

在函数柯里化例子中bind的例子中,柯里化的目的是为了固定可变参数this。而反柯里化,把原来拥有方法的this泛化了,泛化到全部对象均可以借用,也就是替代当前拥有方法的this。

这里,但愿不要混淆,这里的this并非指上例中的this,上例中的this只是由于借用的方法是call。

更有趣的是,unCurrying自己也是方法,它是否能够被借用呢?答案是确定的。这就是js的奇妙之处,反柯里化的奇妙之处。

看下面:

var unCurrying = Function.prototype.unCurrying.unCurrying();
var map = unCurrying(Array.prototype.map);
var sq = map([1, 2, 3],
function(n) {
    return n * n;
});
console.log(sq); // [1,4,9]

不管是柯里化仍是反柯里化,其实反应的都是一种设计思想。这一节先到这里。

相关文章
相关标签/搜索