使用 async/await 是必须避免的陷阱

使用 async/await 是必须避免的陷阱

如果我们使用过 nodejs,那么我们可能已经在 javaSoript 中使用了异步操作。异步任务是一个独立于 JavaSoript 引擎的主线程执行的操作。从本质上讲,这就是应用程序功能没有阻塞的 UI 的原因。

nodejs 的单线程性质,这一点极其重要。

Node.js 利用事件循环来处理所有异步操作,保留了用于计算函数的主线程。

在这里插入图片描述
假设我们对事件循环有相当的了解。在这种情况下,我们会明白,当在调用堆栈中发现一个非同步操作时,JS会把它放到线程池上,线程池将通过 libuv 库异步地执行它。之后 libuv 将执行操作并将其推进到"事件队列"中。"事件队列将被持续监控,事件队列中的事件将被提取并在处理异步操作响应的回调函数上执行。这基本上就是 nodejs 如何处理异步操作。

例如我们可以使用 JavaSrispt 中 Promise 建立异步操作。Promise 返回的一个对象,代表其进程。

// 返回一个Promise对象
function fetchData() {
    return new Promise((resolve, reject) => {
        // 使用setTimeout 模拟一个异步操作
        setTimeout(() => {
            const data = 'Sample Data';
            const success = true; 

            if (success) {
                resolve(data); 
            } else {
                reject('Error: Unable to fetch data');
            }
        }, 2000);
    });
}
const fetchDataPromise = fetchData();
fetchDataPromise.then(data => {
    console.log('Data received:', data);
})
.catch(error => {
    console.error(error);
});

fetchData 方法返回的 Promise 对象包含两种方法:then 和 catch。开发者可以在这两个方法中获取结果。

这使 JavaSrhpt 更加强大,使我们能够构建实时聊天应用程序和API等应用程序。然而,在设计应用程序时,使用JavaSrispt异步操作会有一些常见的缺陷,我们必须考虑这些缺陷,以便能够实现缓解这些问题的方法。

注意:这些陷阱存在于任何 javascript 框架。

回调地狱

使用基于 Promise 的异步操作的关键问题之一是回调地狱。在这种情况下,回调会不断调用 Promise,导致回调链。例如:

function performAsyncOperation(delay: number, message: string): Promise<string> {
  return new Promise((resolve) => {
    setTimeout(() => {
      console.log(message);
      resolve(message);
    }, delay);
  });
}


export async function callbacks() {
  const delay = 1000;
  const message = 'Hello World';
  return performAsyncOperation(delay, message).then((value) => {
    performAsyncOperation(delay, value).then((secondValue) => {
      performAsyncOperation(delay, secondValue).then((thirdValue) => {
        performAsyncOperation(delay, thirdValue).then(() => {
          console.log('End The Callback');
        }).catch(() => {
          console.log('Error');
        });;
      }).catch(() => {
        console.log('Error');
      });;
    }).catch(() => {
      console.log('Error');
    });
  });
}

callbacks() 方法 返回一个 performAsyncOperation 并继续添加更多的异步操作。虽然能在生产中发挥完美的作用。但是,当我们考虑到可维护性时,它将是一个混乱的问题。例如,很难看到什么样的回调应用在什么级别。

所以,我们如何避免这种情况?

为了修复回调地狱问题,我们可以将此转换为 async/await 。所以, 我们看看这个的更新代码 :

function performAsyncOperation(delay: number, message: string): Promise<string> {
  return new Promise((resolve) => {
    setTimeout(() => {
      console.log(message);
      resolve(message);
    }, delay);
  });
}

export async function asyncAwait() {
  try {
    const delay = 1000;
    let message = 'Hello World';

    message = await performAsyncOperation(delay, message);
    message = await performAsyncOperation(delay, message);
    message = await performAsyncOperation(delay, message);

    await performAsyncOperation(delay, message);

    console.log('End The Callback');
  } catch (error) {
    console.log('Error:', error);
  }
}

我们已经成功地将回调地狱重构为更清洁的方法,它使用async/await,这允许我们执行相同的异步代码,而我们在早些时候执行了一个更干净的方法。await 意味着每一行代码在收到回复之前等候。如果它返回一个成功的响应,它将继续到下一个。但是如果它遇到错误,它将跳到公共的catch 整块。这样做可以避免维护多个错误处理程序和使用单个错误处理程序的需要。

同步函数链

我们已经重构我们的代码,使用async/await 块来处理多个异步调用。但是现在,我们可能会注意到这里有一个新问题:

function performAsyncOperation(delay: number, message: string): Promise<string> {
  return new Promise((resolve) => {
    setTimeout(() => {
      console.log(message);
      resolve(message);
    }, delay);
  });
}

export async function issueAsyncAwait() {
  try {
    const delay = 1000;
    let message = 'Hello World';

    await performAsyncOperation(delay, message);
    console.log('Phase 01');
    await performAsyncOperation(delay, message);

    console.log('End The Callback');
  } catch (error) {
    console.log('Error:', error);
  }
}

在这种情况下,我们想执行 console.log(‘Phase 1’) ,但是performAsyncOperation 方法在一个单独的进程中执行,我们的打印应该是在performAsyncOperation 方法执行前完成对吗?

在这里插入图片描述

经过检查,我们可以看到这并不是我们所期待的。怎么回事?

顾名思义,它"等待"整个代码块,直到异步操作返回响应。因此,这使得我们的代码"同步",并创建了一个瀑布调用模式,在这里我们的代码将一个接一个地调用。

因此,如果我们的事件并不相互依赖,如果我们的事件不依赖于非同步操作的输出,我们不必一定要等到非同步操作完成,对吗?

所以,在这种情况下, 考虑使用回调 :

function performAsyncOperation(delay: number, message: string): Promise<string> {
  return new Promise((resolve) => {
    setTimeout(() => {
      console.log(message);
      resolve(message);
    }, delay);
  });
}

export async function asyncAwaitFix() {
  try {
    const delay = 1000;
    let message = 'Hello World';

    performAsyncOperation(delay, message).then((resp) => console.log(`Process the resp: ${resp}.`));
    console.log('Phase 01');
    await performAsyncOperation(delay, message);

    console.log('End The Callback');
  } catch (error) {
    console.log('Error:', error);
  }
}

如你所见,我们重构了performAsyncOperation 方法并使用 .then() 回调。这样做可以让回调作为一个真正的回调执行,并且不会在代码中创建任何"等待"。为了验证我们的理论,让我们检查一下输出:

在这里插入图片描述
如你所见,Phase 01 首先打印了,不再等待到 async 操作完成。

但是要小心使用这个,因为我们可能会创建回调地狱!

循环的性能问题

接下来,让我们谈谈循环。我们都用 JavaScript 写过循环:

 for (let i = 0; i < 5; i++) {
    console.log('Iteration number:', i);
 }

我们循环了一组元素,并对其进行了一些计算。但是如果我们必须在这里执行异步操作呢?假设我们得到了一堆用户身份证。并被要求获取所有身份证的信息(注意:我们的API不支持批量)。 我们可能会这样写:

function getUserInfo(id: number) {
  return new Promise((resolve) => {
    // 模拟异步
    setTimeout(() => {
      resolve({ userId: id, name: `User_${id}`, email: `user${id}@example.com` });
    }, 1000);
  });
}

export async function asyncForLoopIssue(userIds: number[] = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]) {
  const usersInfo: any[] = [];

  for (let i = 0; i < userIds.length; i++) {
    const userInfo = await getUserInfo(userIds[i]);
    usersInfo.push(userInfo);
  }
  console.log({ usersInfo });
  return usersInfo;
}

现在,这个代码再次没有问题。它将按预期在生产中发挥作用。但是,我们被限制在这里的同步循环。这意味着一旦收集到单个用户信息,我们的循环的下一次迭代将开始。因此,这个函数将在10s后执行,并像这样的同步输出:

在这里插入图片描述
这是一个接一个发生的。

但我们该怎么解决?

可以用非线性来执行这个循环:

function getUserInfo(id: number) {
  return new Promise((resolve) => {
    setTimeout(() => {
      resolve({ userId: id, name: `User_${id}`, email: `user${id}@example.com` });
    }, 1000);
  });
}

export async function loopAsyncFix(userIds: number[] = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]) {
  const promises = userIds.map(async (id) => {
    const userInfo = await getUserInfo(id);
    return userInfo;
  })

  const usersInfo = await Promise.all(promises);

  console.log({ usersInfo });

  return usersInfo;
}

现在,这种方法将产生相同的响应。然而,它的实现方式有点不同。

在方法01中,每次迭代都在当前的async操作完成后开始。
async 意味着它应该在不干扰主线程的情况下执行。

第二种方法坚持真正的异步方法,因为它返回最终将执行的 Promise 对象。因此,虽然我们是顺序运行它,但是它将返回随机调用,每个调用都是独立的,并且在没有相关顺序的情况下按自己的速度执行。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/220303.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

华容道问题求解第一部分_思路即方案设计

一、前言 华容道是一种传统的益智游戏&#xff0c;通常由一个长方形木板和若干个方块组成。其中包括一个或多个不同颜色的方块&#xff08;也称为车块&#xff09;和其他大小相同的方块&#xff08;也称为障碍块&#xff09;。游戏的目标是将车块从木板的一个端点移动到另一个…

【mysql】mysgld.log文件太大怎么办

我们有一台测试服务器。跑着一个msyq&#xff0c;发现没有空间了。差看日志文件占用了很多。 怎么破 使用下面命令 echo "" >mysqld.log 执行命令后

PostGIS学习教程九:空间连接

PostGIS学习教程九&#xff1a;空间连接 空间连接&#xff08;spatial joins&#xff09;是空间数据库的主要组成部分&#xff0c;它们允许你使用空间关系作为连接键&#xff08;join key&#xff09;来连接来自不同数据表的信息。我们认为“标准GIS分析”的大部分内容可以表示…

直播预告 | 降本增效持续深化,如何找准 FinOps 关键着力点?

企业落地 FinOps 有哪些实施路径和阶段规划&#xff1f;2023 年&#xff0c;业界 FinOps 取得了哪些进展&#xff1f;12 月 6 日&#xff0c;「降本增效持续深化&#xff0c;如何找准 FinOps 关键着力点」专题直播即将开讲。小红书基础技术部混合云资源管理负责人梁啟成将带来《…

无法从SD卡中删除文件怎么办?

在使用SD卡时&#xff0c;有时我们会无法从SD卡中删除文件&#xff0c;那么这该怎么办呢&#xff1f;下面我们就一起来了解一下吧。 方式1. 检查SD卡&#xff08;读卡器&#xff09;上的写保护选项卡 对于某些SD卡&#xff0c;SD卡的一侧可能有一个开关&#xff0c;并有标有Lo…

AntDesignBlazor示例——创建项目

本示例是AntDesign Blazor的入门示例&#xff0c;在学习的同时分享出来&#xff0c;以供新手参考。 示例代码仓库&#xff1a;https://gitee.com/known/AntDesignDemo 1. 开发环境 VS2022 17.8.2.NET8AntDesign 0.16.2 2. 学习目标 创建新项目安装AntDesign组件包及使用方…

Leetcode 77 组合

题意理解&#xff1a; 给定两个整数 n 和 k&#xff0c;返回范围 [1, n] 中所有可能的 k 个数的组合。 如&#xff1a;n3,k2,则有&#xff1a;12 13 23 一般&#xff0c;我们使用回溯法来解决组合问题。 组合问题没有顺序要求&#xff0c;所以 12 21 是同一个组合&#xff08;如…

【Linux驱动开发】环境搭建Linux驱动开发环境

环境搭建Linux驱动开发环境 1. 简单描述2. 资源3. 安装4. 基本操作和设置 1. 简单描述 基于讯为电子rk3568教程 2. 资源 下载 VMware Workstation Pro 17 链接 Ubuntu 桌面版&#xff08;64位&#xff09; 链接 3. 安装 需要选择自定义硬件&#xff08;内存大于16g 硬盘500g…

试验数字化平台WDP 助力车企数据管理加速度

一 现状 随着现代测控技术的提高&#xff0c;数据结构变得越来越复杂多样&#xff0c;数据量也在日益增大。又因试验条件的限制&#xff0c;大多数企业的数据管理方式主要是通过各类电子文档将试验数据保存在每个工程师的移动电脑中&#xff0c;再进行汇总存储和共享。这种落后…

【计算机系统基石与Linux进程管理深度解析】

​​​​​​​ 【本节重点】 认识冯诺依曼系统 操作系统概念与定位 深入理解进程概念&#xff0c;了解PCB 学习进程状态&#xff0c;学会创建进程&#xff0c;掌握僵尸进程和孤儿进程&#xff0c;及其形成原因和危害 1.冯诺依曼体系结构 我们常见的计算机&#xff0c;如…

电子学会C/C++编程等级考试2022年12月(四级)真题解析

C/C++等级考试(1~8级)全部真题・点这里 第1题:开餐馆 北大信息学院的同学小明毕业之后打算创业开餐馆.现在共有n 个地点可供选择。小明打算从中选择合适的位置开设一些餐馆。这 n 个地点排列在同一条直线上。我们用一个整数序列m1, m2, ... mn 来表示他们的相对位置。由于地…

【深度学习笔记】09 权重衰减

09 权重衰减 范数和权重衰减利用高维线性回归实现权重衰减初始化模型参数定义 L 2 L_2 L2​范数惩罚定义训练代码实现忽略正则化直接训练使用权重衰减 权重衰减的简洁实现 范数和权重衰减 在训练参数化机器学习模型时&#xff0c;权重衰减&#xff08;decay weight&#xff09…

HITOS_LAB5 进程运行轨迹的跟踪与统计

5. 进程运行轨迹的跟踪与统计 5.1. 实验目的 掌握 Linux 下的多进程编程技术&#xff1b;通过对进程运行轨迹的跟踪来形象化进程的概念&#xff1b;在进程运行轨迹跟踪的基础上进行相应的数据统计&#xff0c;从而能对进程调度算法进行实际的量化评价&#xff0c; 更进一步加…

通过时间交织技术扩展ADC采样速率的简要原理

前言 数据采集是将自然界中存在的模拟信号通过模数转换器&#xff08;ADC&#xff09;转换成数字信号&#xff0c;再对该数字信号进行相应的接收和处理。数据采集系统作为数据采集的手段&#xff0c;在移动通信、图向采集、无线电等领域有重要作用。随着电子信息技术的飞速发展…

8_企业架构缓存中间件分布式memcached

企业架构缓存中间件分布式memcached 学习目标和内容 1、能够理解描述网站业务访问流程 2、能够理解网站业务的优化方向 3、能够描述内存缓存软件Memcached的作用 4、能够通过命令行操作Memcached 5、能够操作安装php的memcached扩展 extension 6、能够实现session存储到memcach…

6G的Java软件安装包和2G的Maven仓库分享给大家

文章目录 &#x1f50a;博主介绍&#x1f964;本文内容&#x1f4e2;文章总结&#x1f4e5;博主目标 &#x1f50a;博主介绍 &#x1f31f;我是廖志伟&#xff0c;一名Java开发工程师、Java领域优质创作者、CSDN博客专家、51CTO专家博主、阿里云专家博主、清华大学出版社签约作…

轨道交通故障预测与健康管理PHM系统的应用

轨道交通是现代城市中不可或缺的交通方式&#xff0c;它为人们提供了快速、高效和可靠的出行方式。然而&#xff0c;由于轨道交通系统的复杂性和高负荷运行&#xff0c;设备故障和运营中断问题时有发生。为了提高轨道交通系统的可靠性和安全性&#xff0c;故障预测与健康管理&a…

一文读懂中间件

前言&#xff1a;在程序猿的日常工作中&#xff0c; 经常会提到中间件&#xff0c;然而大家对中间件的理解并不一致&#xff0c;导致了一些不必要的分歧和误解。“中间件”一词被用来描述各种各样的软件产品&#xff0c;在不同文献中有着许多不同的中间件定义&#xff0c;包括操…

花店小程序商城制作攻略教程分享

现如今&#xff0c;随着互联网的快速发展&#xff0c;越来越多的实体店面对客流量不足的问题。特别是对于花店来说&#xff0c;客流量的多少直接影响着销售额和收益。为了解决这一问题&#xff0c;开发一个花店小程序商城成为了不可忽视的选择。 为了开发花店小程序商城&#x…

使用Docker在Debian上构建GRBL模拟器镜像:简明步骤和操作指南

概述编译编写 Dockerfile构建镜像运行测试其他 概述 本文将详细介绍如何在Debian系统上通过Docker构建GRBL模拟器镜像&#xff0c;以便进行数控机床的仿真测试。GRBL是一种开源的控制系统&#xff0c;用于控制三轴CNC机床、激光雕刻、激光切割&#xff0c;而在Docker容器中运…