欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

JS:函数柯里化

程序员文章站 2022-03-15 21:19:10
函数柯里化 1. 柯里化 在计算机科学中,柯里化(Currying)是把接受多个参数的函数变换成接受一个单一参数(最初函数的第一个参数)的函数,并且返回接受余下的参数且返回结果的新函数的技术。 简单来说,就是固定一些参数,返回一个接受剩余参数的函数。 其实就是使用闭包返回一个延迟执行函数。 只看文字 ......

函数柯里化

  1. 柯里化

    在计算机科学中,柯里化(currying)是把接受多个参数的函数变换成接受一个单一参数(最初函数的第一个参数)的函数,并且返回接受余下的参数且返回结果的新函数的技术。

    简单来说,就是固定一些参数,返回一个接受剩余参数的函数。

    其实就是使用闭包返回一个延迟执行函数。

    只看文字描述去理解柯里化可能有点难,举一个很经典的例子:

    // 这个例子是柯里化,也可以说是部分应用
    // 对于什么是部分应用,这里不进行探讨
    function add(num1, num2) {
        return num1 + num2;
    }
    
    function curry(func) {
        let args = [].slice.call(arguments, 1);
    
        return function() {
            let innerargs = [].slice.call(arguments);
            let finalargs = [...args, ...innerargs];
            return func.apply(null, finalargs);
        }
    }
    
    // 得到延迟执行函数
    let curriedadd = curry(add, 5);
    // 可以多次复用得到的函数
    curriedadd(1); // 6
    curriedadd(2); // 7

    看了上面简单的例子,现在来实现能这样使用curriedadd(1)(2, 3)(4)的柯里化函数:

    function aidcurry(func) {
        let args = [].slice.call(arguments, 1);
        return function() {
            return func.apply(null, [...args, ...arguments]);
        }
    }
    
    function curry(func, length) {
        length = length || func.length;
    
        return function() {
            // 传入参数为 0,表示返回结果
            if (arguments.length != 0 && arguments.length < length) {
                let finalargs = [func, ...arguments];
                // 参数的数量不足时,利用闭包将已传入参数存储起来
                return curry(aidcurry.apply(null, finalargs), length - arguments.length);
            } else {
                // 参数数量够了或者中途结束,则返回结果
                return func.apply(null, arguments);
            }
        };
    }
    
    // length 为 4,表示当传入参数到达4个时返回结果
    let curriedadd = curry(function add() {
        return [...arguments].reduce((acc, cur) => acc + cur);
    }, 4);
    
    // 传入参数为4个
    curriedadd(1)(2, 3)(4); // 10
    // 中途结束
    curriedadd(1)(2, 3)(); // 6
    // error: not a function
    curriedadd(1)(2)()(3); // 使用 (),就表示返回结果,已经不是函数
    curriedadd(1)(2, 3)(4)(); // 传入参数已经为4个,所以返回结果了,已经不是函数