Bootstrap

深入理解 JavaScript 中的异步编程

深入理解 JavaScript 中的异步编程

一、引言

在 JavaScript 编程的世界里,异步编程是一个至关重要且充满魅力的概念。随着现代 Web 应用程序的复杂性不断增加,对异步操作的需求也日益凸显。无论是处理用户交互、网络请求还是文件读取,异步编程都发挥着关键作用。它允许我们的程序在等待某些耗时操作完成时,不会阻塞其他代码的执行,从而提高了应用程序的响应能力和性能。在这篇博客中,我们将深入探讨 JavaScript 中的异步编程,包括它的来历、概念、实现方式以及相关的进程与线程知识。

二、异步是什么

(一)简单示例与分析

在 JavaScript 中,异步操作与同步操作有着明显的区别。看下面的代码示例:

// 异步
let a = 1;
setTimeout(() => {
  a += 1;
  console.log('out', a);
}, 1000);
console.log('end', a);
// end 1
// out 2

在这个例子中,setTimeout是一个典型的异步函数。当 JavaScript 引擎执行到setTimeout时,它不会等待setTimeout中的回调函数执行完再继续向下执行,而是直接执行console.log('end', a)。这是因为setTimeout将回调函数放入了一个特殊的队列(我们稍后会详细介绍这个队列),然后 JavaScript 引擎继续执行后续代码。当经过指定的时间(这里是 1000 毫秒)后,回调函数才会被从队列中取出并执行。

再看另一个示例:

// 循环执行
let a = 1;
let timer = setInterval(() => {
  a += 1;
  console.log('out', a);
}, 1000);

setTimeout(() => {
  clearInterval(timer);
  console.log('end', a);
}, 3000);

// out 2
// out 3
// end 3

这里同时使用了setIntervalsetTimeoutsetInterval会每隔 1000 毫秒重复执行一次回调函数,不断地增加a的值并输出。而setTimeout在 3000 毫秒后清除setInterval。在这个过程中,setInterval的执行是异步的,它不会阻塞其他代码的执行。

(二)异步操作背后的机制

对于前面提到的问题,为何eg2interval会执行,而eg1中没有类似的“等待执行”问题呢?这涉及到 JavaScript 的事件循环机制。

当我们使用setTimeoutsetInterval这样的定时器函数时,它们创建的任务被放入了一个定时器队列(这是一个我们看不到的队列)。JavaScript 引擎有一个主执行线程,它按照代码的顺序执行同步代码。当遇到异步函数时,比如setTimeout,它只是安排了一个在未来某个时间执行的任务,然后继续执行后续的同步代码。在主执行线程空闲时,事件循环会检查定时器队列,如果有任务的时间已经到达,就会将任务放入主执行线程执行。setInterval则是不断地在定时器队列中添加任务,只要不被清除,就会按照指定的时间间隔持续添加。

(三)浏览器中的进程与线程

在面试中,常常会问到在浏览器中新开一个窗口是进程还是线程的问题。这涉及到进程和线程的概念。

进程

进程是 CPU 资源分配的基本单位,一个进程包含若干个线程。它与程序有以下区别:

  • 定义不同:进程是动态的,它代表了程序在内存中的一次执行过程,有自己的生命周期,包括创建、就绪、运行、阻塞和结束五种状态。而程序是静态的,是一组指令的集合,存储在磁盘上。
  • 资源分配:同一进程的各线程共享本进程的资源。进程是系统资源分配的基本单位,它拥有独立的内存空间、文件描述符等资源。
  • 状态不同:进程有运行、就绪和阻塞三种基本状态,这些状态反映了进程在 CPU 上的执行情况以及是否在等待某些资源。
线程

线程是进程的执行单元,是 CPU 调度的最小单位,一个进程中可以有多个线程。线程与进程的区别如下:

  • 地址空间和资源:进程间相互独立,每个进程都有自己独立的地址空间和系统资源。而同一进程的各线程共享进程的地址空间和资源,如内存、文件句柄等。
  • 执行过程:同一进程内的线程共享代码段(包括代码、常量、数据结构等),但每个线程有独立的运行栈和程序计数器(PC)。这意味着每个线程都有自己的执行路径和局部变量,但共享进程中的大部分数据。线程是 CPU 调度执行的基本单位,这使得多线程可以在一个进程内并发执行,提高程序的执行效率。

三、异步编程在 JavaScript 中的实现

(一)回调函数

回调函数是 JavaScript 中实现异步编程的一种常见方式。我们已经在setTimeoutsetInterval中看到了回调函数的使用。除了定时器,许多异步操作都使用回调函数。例如,在进行网络请求时:

function handleResponse(data) {
  console.log('Received data:', data);
}

function makeRequest(url, callback) {
  // 这里模拟一个异步网络请求,实际中会使用 XMLHttpRequest 或 fetch 等
  setTimeout(() => {
    const data = { message: 'Hello from server' };
    callback(data);
  }, 2000);
}

makeRequest('https://example.com', handleResponse);
console.log('Request sent.');
// Request sent.
// Received data: { message: 'Hello from server' }

在这个例子中,makeRequest函数接受一个 URL 和一个回调函数。它模拟了一个异步网络请求,在 2000 毫秒后调用回调函数,并传递请求得到的数据。这种方式使得代码可以在请求进行的同时继续执行其他操作,而不会被阻塞。

(二)Promise

Promise 是 JavaScript 中更现代的异步编程解决方案。它提供了一种更优雅的方式来处理异步操作,避免了回调地狱的问题。

function makeRequest(url) {
  return new Promise((resolve, reject) => {
    // 模拟异步请求
    setTimeout(() => {
      const data = { message: 'Hello from server' };
      resolve(data);
    }, 2000);
  });
}

makeRequest('https://example.com')
.then(data => console.log('Received data:', data))
.catch(error => console.log('Error:', error));
console.log('Request sent.');
// Request sent.
// Received data: { message: 'Hello from server' }

在这个Promise示例中,makeRequest函数返回一个Promise对象。当异步操作成功完成时,我们调用resolve函数,将结果传递给then方法中的回调函数。如果出现错误,可以调用reject函数,由catch方法处理错误。Promise的链式调用使得异步代码的逻辑更加清晰。

(三)async/await

async/await是基于Promise的更高级的异步编程语法糖。它使得异步代码看起来更像同步代码,提高了代码的可读性。

async function getData() {
  try {
    const response = await makeRequest('https://example.com');
    console.log('Received data:', response);
  } catch (error) {
    console.log('Error:', error);
  }
}

getData();
console.log('Function called.');
// Function called.
// Received data: { message: 'Hello from server' }

async函数中,我们可以使用await关键字等待一个Promise的结果。这样的代码结构更符合人类的思维方式,让异步编程变得更加容易理解和维护。

四、总结

JavaScript 中的异步编程是现代 Web 开发中不可或缺的一部分。通过理解异步的概念、背后的机制以及各种实现方式,我们能够更好地编写高效、响应迅速的应用程序。从简单的回调函数到更先进的Promiseasync/await,这些技术为我们处理异步操作提供了丰富的工具。同时,了解进程和线程的基础知识有助于我们理解异步操作在浏览器环境中的运行原理。在实际开发中,我们需要根据具体的场景选择合适的异步编程方式,以确保代码的质量和可维护性。希望这篇文章能够帮助你深入理解 JavaScript 中的异步编程,让你在编程之路上更进一步。

无论是构建复杂的单页应用程序还是简单的网页交互,异步编程的知识都将成为你手中的利器,帮助你应对各种挑战,创造出更加流畅和高效的用户体验。如果你有任何问题或者新的见解,欢迎在评论区交流,让我们一起探索 JavaScript 异步编程的更多奥秘。

;