深刻理解 JavaScript 回调函数

做者:Nilesh Sanyal

翻译:疯狂的技术宅javascript

原文:https://dzone.com/articles/ja...前端

未经容许严禁转载java

JavaScript回调函数是成为一名成功的 JavaScript 开发人员必需要了解的一个重要概念。可是我相信,在阅读本文以后,你将可以克服之前使用回调方法遇到的全部障碍。git

在开始以前,首先要确保咱们对函数的理解是扎实的。程序员

快速回顾:JavaScript 函数

什么是函数?

函数是在其中有一组代码的逻辑构件,用来执行特定任务。实际上为了易于调试和维护,函数容许以更有组织的方式去编写代码。函数还容许代码重用。github

你只需定义一次函数,而后在须要时去调用它,而没必要一次又一次地编写相同的代码。面试

声明一个函数

如今,让咱们看看如何在 javascript 中声明一个函数。segmentfault

  1. 使用函数的构造函数: 在这种方法中,函数是在“函数”的构造函数的帮助下建立的。从技术上讲,这种方法比使用函数表达式语法和函数声明语句语法去声明函数的方法效率要低。
  2. 使用函数表达式: 一般这种方法与变量分配相同。简而言之,函数主体被视为一个表达式,而且该表达式被分配给一个变量。使用这种语法定义的函数能够是命名函数或匿名函数。

没有名称的函数被称为匿名函数。匿名函数是自调用的,这意味着它会自动调用起自身。这种行为也称为当即调用的函数表达式(IIFE)。api

  1. 使用函数声明: 这种方法是 JavaScript 中经常使用的老派方法。在关键字“function”以后,你必须指定函数的名称。以后,若是函数接受多个参数或参数,也须要说起它们。虽然这部分是彻底可选的。

    在函数体中,函数必须将一个值返回给调用方。遇到 return 语句后,该函数将会中止执行。在函数内部,参数将会充当局部变量。数组

    一样,在函数内部声明的变量是该函数的局部变量。局部变量只能在该函数内访问,所以具备相同名称的变量能够轻松地用于不一样的函数。

调用一个函数

在下列任何一种状况下,将调用以前声明的函数:

  • 发生事件时,例如,用户单击按钮,或者用户从下拉列表中选择某些选项等等。
  • 从 javascript 代码中调用该函数时。
  • 该函数能够自动调用,咱们已经在匿名函数表达式中进行了讨论。

    () 运算符调用该函数。

什么是回调函数?

按照 MDN 的描述:回调函数是做为参数传给另外一个函数的函数,而后经过在外部函数内部调用该回调函数以完成某种操做

让我用人话解释一下,回调函数是一个函数,将会在另外一个函数完成执行后当即执行。回调函数是一个做为参数传给另外一个 JavaScript 函数的函数。这个回调函数会在传给的函数内部执行。

在 JavaScript 中函数被看做是一类对象。对于一类对象,咱们的意思是指数字、函数或变量能够与语言中的其余实体相同。做为一类对象,能够将函数做为变量传给其余函数,也能够从其余函数中返回这些函数。

能够执行这种操做的函数被称为高阶函数。回调函数其实是一种模式。 “模式”一词表示解决软件开发中常见问题的某种行之有效的方法。最好将回调函数做为回调模式去使用。

为何咱们须要回调

客户端 JavaScript 在浏览器中运行,而且浏览器的主进程是单线程事件循环。若是咱们尝试在单线程事件循环中执行长时间运行的操做,则会阻止该过程。从技术上讲这是很差的,由于过程在等待操做完成时会中止处理其余事件。

例如,alert 语句被视为浏览器中 javascript 中的阻止代码之一。若是运行 alert,则在关闭 alert 对话框窗口以前,你将没法在浏览器中进行任何交互。为了防止阻塞长时间运行的操做,咱们使用了回调。

让咱们深刻研究一下,以便使你准确了解在哪一种状况下使用回调。

image.png

在上面的代码片断中,首先执行 getMessage()函数,而后执行 displayMessage() 。二者都在浏览器的控制台窗口中显示了一条消息,而且都当即执行。

在某些状况下,一些代码不会当即执行。例如,若是咱们假设 getMessage() 函数执行 API 调用,则必须将请求发送到服务器并等待响应。这时咱们应该如何处理呢?

如何使用回调函数

我认为与其告诉你 JavaScript 回调函数的语法,不如在前面的例子中实现回调函数更好。修改后的代码段显示在下面的截图中。

image.png

为了使用回调函数,咱们须要执行某种没法当即显示结果的任务。为了模拟这种行为,咱们用 JavaScript 的 setTimeout() 函数。该函数会暂停两秒钟,而后在控制台窗口中显示消息“ Hi,there”。

“显示的消息”将被显示在浏览器的控制台窗口中。在这种状况下,首先,咱们须要等待 getMessage() 函数。成功执行此函数后,再执行 displayMessage() 函数。

回调的工做方式

让我解释一下前面的例子在幕后发生的事。

从上一个例子能够看到,在 getMessage() 函数中,咱们传递了两个参数。第一个参数是 msg 变量,该变量显示在浏览器的控制台窗口中,第二个参数是回调函数。

如今,你可能想知道为何将回调函数做为参数进行传递 —— 要实现回调函数,咱们必须将一个函数做为参数传给另外一个函数。

getMessage() 完成任务后,咱们将调用回调函数。以后,当调用 getMessage() 函数时,将引用传给displayMessage() 函数,该函数就是回调函数。

注意,当调用 getMessage() 函数时,咱们仅将其引用传给 displayMessage() 函数。这就是为何你不会在它旁边看到函数调用运算符,也就是() 符号。

Javascript 回调是异步的吗?

JavaScript 被认为是单线程脚本语言。单线程是指 JavaScript 一次执行一个代码块。当 JavaScript 忙于执行一个块时,它不可能移到下一个块。

换句话说,咱们能够认为 JavaScript 代码本质上老是阻塞的。可是这种阻塞性使咱们没法在某些状况下编写代码,由于在这些状况下咱们没有办法在执行某些特定任务后当即获得结果。

我谈论的任务包括如下状况:

  • 经过对某些端点进行 API 调用来获取数据。
  • 经过发送网络请求从远程服务器获取一些资源(例如,文本文件、图像文件、二进制文件等)。

为了处理这些状况,必须编写异步代码,而回调函数是处理这些状况的一种方法。因此从本质上上说,回调函数是异步的。

Javascript 回调地狱

当多个异步函数一个接一个地执行时,会产生回调地狱。它也被称为厄运金字塔。

假设你要获取全部 Github 用户的列表。而后在用户中搜索 JavaScript 库的主要贡献者。再而后,你想要在用户中获取姓名为 John 的人员的详细信息。

为了在回调的帮助下实现这个功能,代码应该以下所示:

http.get('https://api.github.com/users', function(users) {
  /* Display all users */
  console.log(users);
  http.get('https://api.github.com/repos/javascript/contributors?q=contributions&order=desc', function(contributors) {
  /* Display all top contributors */
    console.log(contributors);
    http.get('https://api.github.com/users/Jhon', function(userData) {
    /* Display user with username 'Jhon' */
      console.log(userData);
    });
  });
});

从上面的代码片断中,你能够看到代码变得更加难以理解,以及难以维护和修改。这是由回调函数的嵌套而引起的。

如何避免回调地狱?

可使用多种技术来避免回调地狱,以下所示。

  1. 使用promise
  2. 借助 async-await
  3. 使用 async.js 库

使用 Async.js 库

让咱们谈谈怎样用 async.js 库避免回调地狱。

根据 async.js 官方网站的描述:Async 是一个工具模块,它提供了直接、强大的函数来使用异步 JavaScript

Async.js 总共提供约 70 个函数。如今,咱们将仅讨论其中两个,即 async.waterfall()async.series()

async.waterfall()

当你要一个接一个地运行某些任务,而后将结果从上一个任务传到下一个任务时,这个函数很是有用。它须要一个函数“任务”数组和一个最终的“回调”函数,它会在“任务”数组中全部的函数完成后,或者用错误对象调用“回调”以后被调用。

var async = require('async');
async.waterfall([
    function(callback) {
      /*  
        Here, the first argument value is null, it indicates that
        the next function will be executed from the array of functions.
        If the value was true or any string then final callback function
        will be executed, other remaining functions in the array 
        will not be executed.
      */
        callback(null, 'one', 'two');
    },
    function(param1, param2, callback) {
        // param1 now equals 'one' and param2 now equals 'two'
        callback(null, 'three');
    },
    function(param1, callback) {
        // param1 now equals 'three'
        callback(null, 'done');
    }
], function (err, result) {
    /*
      This is the final callback function.
      result now equals 'done'
    */
});

async.series()

当你要运行一个函数而后在全部函数成功执行后须要获取结果时,它颇有用。 async.waterfall()async.series() 之间的主要区别在于, async.series() 不会将数据从一个函数传递到另外一个函数。

async.series([
    function(callback) {
        // do some stuff ...
        callback(null, 'one');
    },
    function(callback) {
        // do some more stuff ...
        callback(null, 'two');
    }
],
// optional callback
function(err, results) {
    // results is now equal to ['one', 'two']
});

Javascript 回调与闭包

闭包

用技术术语来讲,闭包是捆绑在一块儿的函数的组合,引用了其周围的状态。

简而言之,闭包容许从内部函数访问外部函数的做用域。

要使用闭包,咱们须要在一个函数内部定义另外一个函数。而后,咱们须要将其返回或传给另外一个函数。

回调

从概念上讲,回调相似于闭包。回调基本上是把一个函数做为另外一个函数的用法。

最后的话

但愿本文能消除你对 javascript 回调函数的全部疑问。若是你以为这篇文章有帮助,请与他人分享。


本文首发微信公众号:前端先锋

欢迎扫描二维码关注公众号,天天都给你推送新鲜的前端技术文章

欢迎扫描二维码关注公众号,天天都给你推送新鲜的前端技术文章

欢迎继续阅读本专栏其它高赞文章:


相关文章
相关标签/搜索