所有 javascript 回调都是异步的吗?如果不是,我怎么知道哪些是?

IT技术 javascript node.js asynchronous synchronous
2021-01-23 00:37:04

我很好奇所有 javascript 回调是否都是异步的,或者是否仅在某些情况下是这种情况。另外,我确定是什么使 javascript 代码异步(或使用异步 javascript 的方法)在浏览器和 nodejs 之间有所不同,所以我想知道在每种情况下什么构成真正的异步 javascript。

我的印象是,在以下场景中,我实际上并不是在编写异步代码。

function addOne(value){
  value = value + 1;
  return value;
}

function simpleMap(values, callback){
  for(i = 0; i < values.length; i++){
    val = values[i];
    val = callback(val);
    values[i] = val;
  }
  return values;
}

newValues = simpleMap([1,2,3], addOne);

但是,例如,我知道 jQuery 的 AJAX 函数是真正异步的(不考虑现在可用的Promise)。是什么让 jQuery 的 AJAX 异步?是不是就这么简单,涉及到XHR请求,在浏览器中,所有XHR请求都是异步的?

我对 nodejs 环境有同样的问题。如果 node 中的某些东西涉及文件 i/o、process.nextTick、setTimeout 或 setInterval 之类的东西,它是否只能是异步的?为什么当我使用 mongodb/mongoose 执行数据库调用之类的操作时,这是异步的吗?幕后到底发生了什么事情才如此?

异步“情况”是否由环境预先确定?或者有什么方法可以让自己的功能真正异步而不利用环境的非常具体的功能(例如xhr,节点中的文件io,process.nexttick等)?

5个回答

我很好奇是否所有的 javascript 回调都是异步的

不。例如, 使用的回调Array#sort不是异步的,也不是使用的回调String#replace

您知道回调是否异步的唯一方法是从其文档中。通常,涉及对外部资源(例如 ajax 调用)的请求是异步的,而其他的可能是也可能不是。

但是,例如,我知道 jQuery 的 AJAX 函数是真正异步的......

不一定,因为目前 jQuery 仍然具有async您可以设置false以强制同步请求标志(这不是一个好主意,他们将删除它,但您可以。jQuery 将标志传递给提供同步/异步行为的底层浏览器对象。)

是什么让 jQuery 的 AJAX 异步?

浏览器。jQuery 的ajax 调用使用XMLHttpRequest对象(或在某些情况下,一个script元素),默认为浏览器提供的异步操作。

或者有什么方法可以使自己的功能真正异步而不利用环境的非常特定的功能......

直到最近,没有。直到第 5 版规范,JavaScript语言基本上对线程和异步性的整个概念保持沉默;只有当你进入环境时,它才会出现。使某些事情异步的唯一方法是使用主机提供的函数,例如nextTick(或任何异步完成的各种操作)在 NodeJS 或setTimeout浏览器上。

在 2015 年 6 月的 ECMAScript 第 6 版规范中,他们在语言中引入了promise回调函数通过迷上了一个ES6的Promisethen,并如总是异步调用(即使回调时附加Promise中已经解决),所以JavaScript有不同步性在现在的语言水平。因此,如果您实现函数使其返回Promise而不是接受回调,您就会知道then与其挂钩回调将被异步触发。

您自己调用的回调是常规函数调用,它们始终是同步的。

某些本机 API(例如 AJAX、地理定位、Node.js 磁盘或网络 API)是异步的,将在稍后的事件循环中执行它们的回调。

如果您从异步回调中同步调用回调,它最终也会是异步的。

要创建自己的异步函数,您必须使用解释器可能提供的其他异步函数。

例如,此代码定义了一个异步函数“addKeyHandler”。但这只是因为 document.onKey 被 JS 引擎异步调用。JavaScript 引擎能够提供异步功能,因为操作系统提供了这样的功能,然后由 JS 使用。反过来,操作系统只能提供异步功能,因为硬件提供了它(称为硬件中断)。

但是,如果操作系统和硬件不提供任何异步功能,则仍然可以编写 JS 解释器。但它必须使用无限循环并检查每次迭代是否发生任何事件,然后调用适当的回调。这意味着 CPU 将始终处于满载状态。

var keyCallbacks = [];

var addKeyHandler = function(f) {
  keyCallbacks.push(f);
};

document.onkeypress = function(e) {
  keyCallbacks.forEach(function(f) {
      f(e);
  });
};

addKeyHandler(function(e) {
    console.log(String.fromCharCode(e.charCode));
});
很好的答案!您在回调、操作系统和硬件之间所做的相似之处更容易理解。
2021-04-04 00:37:04

简单地进行回调不会使函数异步。有许多函数示例采用函数参数但不是异步的,例如Array's forEach

对于异步函数,它需要执行异步操作。引入异步性的方法可以是

  • 定时器功能setTimeoutsetInterval
  • 特殊功能nextTicksetImmediate
  • 执行 I/O(监听网络、查询数据库、读取或写入资源)
  • 订阅事件

根据文章“采用回调会使函数异步吗?”

想要一个简单的答案:

除非您正在处理 Promise,否则 JS 回调只有在依赖于 JS 外部的 API(例如由浏览器提供)时才是异步的。

setTimeout, fetch, 等是异步的,因为它们依赖于外部 api。例如,它是 web api 的setTimeout一部分windowOrGlobalWorker,并且fetch本身就是一个 web api。)

如果回调不依赖于外部 API,则它是同步的。

Promise 是 JS 中唯一的原生异步功能。

了解这一切的一个好方法是至少阅读 MDN 入门中关于异步的前几篇文章:https : //developer.mozilla.org/en-US/docs/Learn/JavaScript/Asynchronous