最新公告
  • 欢迎您光临起源地模板网,本站秉承服务宗旨 履行“站长”责任,销售只是起点 服务永无止境!立即加入钻石VIP
  • 函数柯里化

    正文概述 掘金(油头周大锅)   2020-12-11   431
    函数柯里化(Currying)
    这样做有什么好处 ?
    参数复用实例
    //普通封装
    
    function check(reg, txt) {
      return reg.test(txt);
    }
    console.log(check(/\d+/g, "test1"));
    // =>true
    console.log(check(/\d+/g, "testtest"));
    // =>false
    
    //柯里化封装
    
    function curryingCheck(reg) {
      return function (txt) {
        return reg.test(txt);
      };
    }
    
    let hasNumber = curryingCheck(/\d+/g);
    let hasLetter = curryingCheck(/[a-z]+/g);
    
    //校验
    console.log(hasNumber("test1"));
    // =>true
    console.log(hasNumber("testtest"));
    // =>false
    console.log(hasLetter("21212"));
    // =>false
    //校验
    console.log(curryingCheck(/\d+/g)("asda1"));
    // =>true
    console.log(curryingCheck(/\d+/g)("asda"));
    // =>false
    

    参数复用,利用的事闭包原理,让前面传输过来的参数不要被释放掉。

    参数复用
    /**
     * 延迟计算
     */
    const add = (...args) => args.reduce((a, b) => a + b);
    
    function curryingSum(func) {
      const args = [];
      return function result(...rest) {
        if (rest.length === 0) {
          return func(...args);
        } else {
          args.push(...rest);
          return result;
        }
      };
    }
    

    这个函数当参数为空的时候执行了内部参数所有值的相加,当参数不为空的时候将缓存起来, 在为空的时候再相加,同样是用闭包的方式来实现。

    提前确认

    这一特性经常是用来对浏览器的兼容性做出一些判断并初始化api,比如说我们目前用来监听事件大部分情况是使用addEventListener来实现的,但是一些较久的浏览器并不支持该方法,所以在使用之前,我们可以先做一次判断,之后便可以省略这个步骤了。

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

    通用的封装方法
    function curryingFun(fn, length) {
      //第一次调用获取函数 fn 参数的长度,后续调用获取 fn 剩余参数的长度
      length = length || fn.length;
      // curryingFun 包裹之后返回一个新函数,接收参数为 …args
      return function (...args) {
        // 新函数接收的参数长度是否大于等于 fn 剩余参数需要接收的长度
        return args.length >= length
          ? // 满足要求,执行 fn 函数,传入新函数的参数
            fn.apply(this, args)
          : // 不满足要求,递归 curryingFun 函数,新的 fn 为 bind 返回的新函数(bind 绑定了…args 参数,未执行),新的 length 为 fn 剩余参数的长度
            curryingFun(fn.bind(this, ...args), length - args.length);
      };
    }
    

    ES6写法

    const curryingFun = (fn, arr = []) => (...args) =>
      ((arg) => (arg.length === fn.length ? fn(...arg) : curryingFun(fn, arg)))([
        ...arr,
        ...args,
      ]);
    

    检测一下

    const fn = curryingFun(function (a, b, c) {
      console.log([a, b, c]);
    });
    fn("a", "b", "c"); // ["a", "b", "c"]
    
    const fn2 = curryingFun(function (a, b, c) {
      console.log(a + b + c);
    });
    fn2(1)(2)(3); .//6
    

    关于柯里化性能方面
    • 存取arguments对象通常要比存取命名参数要慢一点
    • 一些老版本的浏览器在arguments.length的实现上是相当慢的
    • 使用fn.apply( … ) 和 fn.call( … )通常比直接调用fn( … ) 稍微慢点
    • 创建大量嵌套作用域和闭包函数会带来花销,无论是在内存还是速度上

    关于柯里化的意义

    把函数完全变成「接受一个参数;返回一个值」的固定形式,利用闭包的特性,去除多余参数传递,提高可读性以及代码质量。


    参考文档
    • 深入高阶函数应用之柯里化<木易杨> www.imooc.com/article/291…

    • 详解JS函数柯里化 flowsands www.jianshu.com/p/2975c25e4…


    起源地下载网 » 函数柯里化

    常见问题FAQ

    免费下载或者VIP会员专享资源能否直接商用?
    本站所有资源版权均属于原作者所有,这里所提供资源均只能用于参考学习用,请勿直接商用。若由于商用引起版权纠纷,一切责任均由使用者承担。更多说明请参考 VIP介绍。
    提示下载完但解压或打开不了?
    最常见的情况是下载不完整: 可对比下载完压缩包的与网盘上的容量,若小于网盘提示的容量则是这个原因。这是浏览器下载的bug,建议用百度网盘软件或迅雷下载。若排除这种情况,可在对应资源底部留言,或 联络我们.。
    找不到素材资源介绍文章里的示例图片?
    对于PPT,KEY,Mockups,APP,网页模版等类型的素材,文章内用于介绍的图片通常并不包含在对应可供下载素材包内。这些相关商业图片需另外购买,且本站不负责(也没有办法)找到出处。 同样地一些字体文件也是这种情况,但部分素材会在素材包内有一份字体下载链接清单。
    模板不会安装或需要功能定制以及二次开发?
    请QQ联系我们

    发表评论

    还没有评论,快来抢沙发吧!

    如需帝国cms功能定制以及二次开发请联系我们

    联系作者

    请选择支付方式

    ×
    迅虎支付宝
    迅虎微信
    支付宝当面付
    余额支付
    ×
    微信扫码支付 0 元