前言:

本篇主要是介绍 JavaScript使用函数的高级方法,函数是JavaSCript中最有趣的部分,利用Function特性可以编写出很多非常有意思的代码,本篇主要包括:函数回调,高阶函数以及函数柯里化。

 
1.   函数回调
     对于异步编程JavaScript API如SetTimeout或者JQuery的$.ajax又或者越来越火的Node.js编程,在事件循环上都是非阻塞的,意思就是,当你使用异步非阻塞API的使用,调用在回调函数执行之前立即返回,而这些回调在不久之后执行。
 setTimeout(function(){
console.log("hello");
},2000)
// ... 大概2s之后(实际上大于2s)
// => hello

Node中一个非阻塞代码实例

创建一个文件 input.txt ,内容如下:

 “Hello World” 

创建 main.js 文件, 代码如下:

 var fs = require("fs");
// 封装一个readFile函数
var readFile=function(filename,callback){
fs.readFile('input.txt', function (err, data) {
if (err) return callback(err);
return callback(null,data);
});
}
//调用
readFile('input.txt',function(err,data){
if(err)
handle(err);
else
handle(data);
}
我们不需要等待文件读取完,这样就可以在读取文件时同时执行接下来的代码,由回调函数执行处理读取的文件内容,大大提高了程序的性能。

JQuery的XMLHttpRequest也是利用回调执行异步编程

 $.get( "example.php", function() {
alert( "success" );
})
.done(function() {
alert( "second success" );
})
.fail(function() {
alert( "error" );
})
.always(function() {
alert( "finished" );
});
 
2.   高阶函数
     高阶函数可以分为:
  1. 以其他函数为参数的函数
  2. 返回其他函数的函数
    这一部分更偏向于JavaScript函数式编程的部分,但是利用高阶函数可以构建功能更强大的函数,所以在这里讲了一下。
 
  1. 以其他函数为参数的函数
    
       其实,回调函数也是属于这种模式,利用这种模式,我们可以实现Model和Action层的逻辑分离,举个例子吧
       我们利用数据库做了一个小型的登录用户检查,添加和删除。   
    •    建立一个 model/User.js 主要负责和数据库进行数据的增删查改
 var mysql = require('mysql');
var pool = mysql.createPool({
// 登录数据库的配置
});
function addUser(param,callback) {
pool.getConnection(function(err, connection) {
if(err) return callback(err);
var $sql="INSERT INTO user(id, password) VALUES(?,?)";
connection.query( $sql, [param.id, param.password],function(err, result) {
if(err)
return callback(err);
else
connection.release();
return callback(null,result);
});
});
}
    • 通过传入添加数据和一个回调函数作为参数,我们在视图层可以自定义对添加失败和成功的操作,而无需理会和数据库的具体操作,当数据库发生变化的时候,只需要修改model的User.js,实现外模式和模式的逻辑独立性
 

2. 返回其他函数的函数

          这里我们将了解返回函数和闭包的高阶函数
          举个例子,我们创建一个makeAdder的高阶函数,他的参数配置了其返回函数每次添加数值的大小

 function makeAdder(addVal){
return function(arg){
return addVal+arg;
}
}
var add2=makeAdder(2);
var a=add2(10)
console.log(a);
// => 12
 
按照这样子我们可以自由配置 add10,add5等等。。。
[注]: 其实这里运用了闭包的知识,我们都知道闭包会保存外部函数的作用域链,可以捕获外部函数的参数;当我们返回一个add2()函数的时候,这个函数会一直记住外部函数的addVal的值
[缺点]:过度使用闭包,会造成内存泄漏,就像上面的add2()函数,会一直保存着外部函数的作用域链,直到自己被销毁
 
 
3.   函数柯里化
 
         柯里化函数为每一个逻辑参数返回一个新函数。函数柯里化的基本方法和函数绑定是一样的:使用一个闭包返回一个函数。
        其实上面的makeAdder()其实也可以算是一种柯里化函数,返回的是一个函数,但从技术上讲,并不算是真正意义上的柯里化函数,我们来看一个真正意义的柯里化函数例子:

 function curry(fn){
  var args=[].slice.call(arguments,1);
  return function(){
  var innerArgs=[].slice.call(arguments);
  var finalArgs=args.concat(innerArgs);
  return fn.apply(null,finalArgs);
  }
}
 function add(){
var args=Array.prototype.slice.call(arguments);
return args.reduce(function(prev,next){
return prev+next;
});
}
 var curryAdd=curry(add);
var result=curryAdd(1,2,3);
console.log(result);
// => 6
var curryAdd2=curry(add,2);
var result=curryAdd2(2,3);
console.log(result);
// => 7
两种方式都可以,第二个函数可以代表一个 加2函数
curry()函数的主要工作就是将返回函数的参数进行排序。curry()的第一个参数是要进行柯里化的函数,其他参数是要传入的值。利用[].slice.call(arguments,1) 获取外部函数的参数;在内部函数中,使用innerArgs存放所有传入的参数,最后使用concat将所有参数组合成finalArgs,利用apply()将参数传递给fn函数。
[注意]:这个函数没有考虑执行环境,所以调用apply()的第一个参数为null。
[缺点]:利用柯里化函数可以实现复杂的算法和功能,但是都不可以滥用,因为他们是基于闭包实现的,会带来一定的开销。
 
 
 
 
 
 
 
 
 
 
 
 

JavaScript系列:高级函数篇的更多相关文章

  1. 深入理解JavaScript系列(结局篇)

    介绍 最近几个月忙得实在是不可开交,终于把<深入理解JavaScript系列>的最后两篇“补全”了,所谓的全是不准确的,因为很多内容都没有写呢,比如高性能.Ajax安全.DOM详解.Jav ...

  2. 30s源码刨析系列之函数篇

    前言 由浅入深.逐个击破 30SecondsOfCode 中函数系列所有源码片段,带你领略源码之美. 本系列是对名库 30SecondsOfCode 的深入刨析. 本篇是其中的函数篇,可以在极短的时间 ...

  3. javascript系列-class4.函数

    欢迎加入前端交流群来py: 转载请标明出处!                   在火影的世界中存在忍术,忍术是把强大的能量集中在一起以各种各样不同的形式发射出来.怎样使用各种各样的忍术那?通过结印. ...

  4. JavaScript系列:函数 自执行 表达式 声明 定义

    可用方式 (function($) {})(jQuery); !function( $ ){}(jQuery); +function( $ ){}(jQuery); -function( $ ){}( ...

  5. javascript之高级函数应用思想

    1.级联函数:应用对象方法调用的连写 function A(){ this.a = ''; this.b = ''; this.c = ''; } //改造一下 A.prototype = { A.p ...

  6. javaScript的高级函数

    1.map() map()方法返回一个新数组,新数组是原始数组调用函数之后处理后的值. map()方法按照原始数组元素顺序依次处理元素. map不会对空数组进行检测. map不会改变原始数组 . 参数 ...

  7. Oracle高级函数篇之递归查询start with connect by prior简单用法

    路飞:" 把原来CSDN的博客转移到博客园咯!" 前段时间,自己负责的任务中刚好涉及到了组织关系的业务需求,自己用了oracle递归查询.下面简单来举个例子.在工作中我们经常会遇到 ...

  8. 深入理解JavaScript系列

    转自http://www.cnblogs.com/TomXu/archive/2011/12/15/2288411.html 深入理解JavaScript系列(1):编写高质量JavaScript代码 ...

  9. 深入理解JavaScript系列(转自汤姆大叔)

    深入理解JavaScript系列文章,包括了原创,翻译,转载,整理等各类型文章,如果对你有用,请推荐支持一把,给大叔写作的动力. 深入理解JavaScript系列(1):编写高质量JavaScript ...

随机推荐

  1. Count on a tree 树上区间第K小

    Count on a tree 题意:求路径 u到v上的 第k小的权重. 题解:先DFS建数, 然后对于每个节点往上跑出一颗主席树, 然后每次更新. 查询的时候, u, v, k, 找到  z = l ...

  2. HDU 1003 Max Sum * 最长递增子序列(求序列累加最大值)

    Max Sum Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Sub ...

  3. sed一些常用命令

    [转] http://blog.chinaunix.net/uid-20754793-id-177657.html 下面是我学习sed时参照参考书总结的一些常用sed命令,基本上每条语句都进行了调试1 ...

  4. 2018 Multi-University Training Contest 1(部分题解)

    Maximum Multiple Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  5. 一次搞懂建模语言UML

    Unified Modeling Language (UML)又称统一建模语言或标准建模语言,它是一个支持模型化和软件系统开发的图形化语言,为软件开发的所有阶段提供模型化和可视化支持,包括由需求分析到 ...

  6. MongoDB的入门使用以及遇到的坑

    一:MonoDB的简单介绍 MongoDB是一个介于关系型数据库与非关系型数据库中间的数据库,是使用C++进行编写的,他的优点是在支持的查询格式特别的强大,可以进行存储比较复杂的数据类型,支持建立索引 ...

  7. 玩转 Springboot 2 之热部署(DevTools)

    Devtools 介绍 SpringBoot 提供了热部署的功能,那啥是热部署累?SpringBoot官方是这样说的:只要类路径上的文件发生更改,就会自动重新启动应用程序.在IDE中工作时,这可能是一 ...

  8. Scratch Blocks本地环境搭建

    关于Scratch Blocks环境的搭建,大家在实现的过程中还是有很多的问题,目前谷歌和MIT的工程师也在进一步完善.可以通过以下方式,简单快捷的导出Scratch Blocks对应的index.h ...

  9. “真”pandas“假”sql

    这篇博客利用了 pandas 对数据像 sql 一样去处理. 读取测试数据 import pandas as pd import numpy as np url = 'https://raw.gith ...

  10. 并发编程之线程创建到销毁、常用API

    在前面一篇介绍了线程的生命周期[并发编程之多线程概念],在本篇将正式介绍如何创建.中断线程,以及线程是如何销毁的.最后,我们会讲解一些常见的线程API. 线程创建 Java 5 以前,实现线程有两种方 ...