当前位置: 首页 > news >正文

JavaScript异步编程——05-回调函数

我们在前面的文章《JavaScript 基础:异步编程/单线程和异步》中讲过,Javascript 是⼀⻔单线程语⾔。早期我们解决异步场景时,⼤部分情况都是通过回调函数来进⾏。

(如果你还不了解单线程和异步的概念,可以先去回顾上一篇文章。)

回调函数的定义

把函数 A 传给另一个函数 B 调用,那么函数 A 就是回调函数。

例如在浏览器中发送 ajax 网络请求,或者在定时器中执行异步任务,就是最常⻅的异步场景。发送请求后,需要等待一段时间,等服务端响应之后我们才能拿到结果。如果我们希望等待异步任务结束之后再执⾏想要的操作,就只能通过回调函数这样的⽅式进⾏处理。

  
const dynamicFunc = function (callback) {setTimeout(function () {console.log("一开始在这里执行异步任务 task1,延迟3秒执行");// task1: total 计数let total = 0;for (let i = 0; i < 10; i++) {total += i;}​// 等待异步任务 task1 执行完成后,通过回调传入的 callback() 函数,通知外面的调用者,可以开始做后续任务 task2 了// 如果有需要的话,可以把 task1 的执行结果 total 传给外面。callback && callback(total);}, 3000);};​// 执行同步任务 task2。需要先等 异步任务 task1做完。dynamicFunc(function (value) {console.log("外面监听到,异步任务 task1已经完成了,并且还能拿到 task1的执行结果 value");console.log("task1的返回值value:" + value);​// task2:将task1的执行结果乘以2const result = value * 2;console.log("result:" + result);});

上⾯的例⼦中,dynamicFunc() 函数里面的 setTimeout()就是⼀个异步函数,在里面执行了一些异步任务,延迟3秒执行。dynamicFunc() 的参数 callback() 就是一个回调函数。这段代码的诉求是:先等待 异步任务 task1 做完,再做 同步任务task2。我们来分析一下。

已知异步任务 task1 需要3秒才能做完。3秒结束后,通知 dynamicFunc 函数的调用者,里面的异步任务 task1 已经做完了,外面可以开始做后续的任务 task2 了。那要怎么通知呢?在ES5中,最常见的做法就是需要回调传入的 callback 函数(也就是回调函数), 通知外面的调用者。并且,如果有需要的话,外面还可以拿到异步任务task1的执行结果 total(详见代码注释)。

(注:callback这个单词并不是关键字,可以自由命名,我们通常习惯性地用“回调”这个词的英文名 callback 代表回调函数。)

回调函数的异常处理

实际开发中,为什么会经常存在异步任务呢?这是因为,有很多函数在执行时无法立即完成,我们也不知道它什么时候能完成。但是,我们需要等待它完成后,才能做接下来的事情。换句话说,我们接下来要做的事情,需要依赖此前的异步任务。

比如, ajax 网络请求就是典型的异步任务。在渲染一个页面时,我们需要请求接口,获取页面所需要的数据。等接口请求完成、数据准备好之后,前端就可以对数据进行处理,并将数据渲染到页面了。前端做的这部分事情,就是在回调函数里面做。

当然,异步任务在执行时可能出现异常、错误信息、执行失败等等。当出现异常时,往往导致后续的回调函数无法执行。这就需要在异步任务中将异常信息通知给外部。

代码举例如下:

 
// 封装异步任务const dynamicFunc = function (number, successCallback, failureCallback) {setTimeout(function () {console.log('一开始在这里执行异步任务 task1,延迟3秒执行');let total = 0;for (let i = 0; i < 10; i++) {total += i;}if (number > 0) {// 异步任务执行成功successCallback && successCallback(total);} else {// 异步任务执行鼠标failureCallback && failureCallback('异步任务执行失败');}}, 3000);};​// 执行异步任务:等待 异步任务 执行完成后,再执行回调函数。dynamicFunc(100,(value) => {console.log('异步函数调用成功:' + value);// task2:将task1的执行结果乘以2const result = value * 2;console.log('result:' + result);},(err) => {console.log('异步函数调用失败:', err);});

处理异步任务的基本模型

我们以“发送网络请求”为例,通过回调函数处理异步任务时,既有请求成功的情况,也有请求失败的情况。其基本处理模型如下:

(1)调用一个异步函数,在这个函数中发送网络请求(也可以用定时器来模拟异步任务)。

(2)如果网络请求成功,则告知调用者请求成功,并将接口返回的数据传递出去。

(3) 如果网络请求失败,则告知调用者发送失败,并将错误信息传递出去。

ES5中,回调函数处理异步任务的基本代码结构如下:

 // ES5中,使用传统的回调函数,处理异步任务的基本模型​// 封装异步任务function requestData(url, successCallback, failureCallback) {const res = {retCode: 0,data: 'qiangu yihao`s data',errMsg: 'network is error',};setTimeout(() => {if (res.retCode == 0) {// 网络请求成功successCallback(res.data);} else {// 网络请求失败failureCallback(res.errMsg);}}, 1000);}​// 调用(请求)异步任务requestData('www.qianguyihao.com/xxx',// 成功监听res => {console.log('异步任务执行成功:', res);},// 失败监听err => {console.log('异步任务执行失败:', err);});

我们一定要记住这个处理模型,它我们学习异步编程的范式之一。如果前端接下来要做的事情需要依赖这个异步任务、需要等待这个异步任务做完之后才能继续,那就符合上面的处理模型。

ES5中,回调的缺点(异步代码的困境)

上面的回调函数的写法,都是ES5的写法。ES5中回调的写法比较直观,不需要 return,层层嵌套即可。但也存在两个问题:

    1. 如果嵌套过深,则会出现回调地狱的问题。

    1. 不同的异步函数,回调的参数,在写法上可能不一致,导致不规范、且需要单独记忆

我们来具体看看这两个问题。

1、回调地狱的问题

如果多个异步任务存在依赖关系(比如,需要等第一个异步任务执行完成后,才能执行第二个异步函数;等第二个异步任务执行完毕后,才能执行第三个异步任务),就需要多个异步任务进⾏层层嵌套,⾮常不利于后续的维护,而且会导致回调地狱(callback hell)的问题。

简而言之,当一个回调函数嵌套另一个回调函数时,就会出现一个嵌套结构。如果嵌套次数过多,就会出现回调地狱的情况。像下面这样:

img

关于回调地狱,我们来举一个形象的例子:

假设买菜、做饭、洗碗、倒厨余垃圾都是异步的。

但真实的场景中,实际的操作流程是:买菜成功之后,才能开始做饭。做饭成功后,才能开始洗碗。洗碗完成后, 再倒厨余垃圾。这里的一系列动作就涉及到了多层嵌套调用,也就是回调地狱。

关于回调地狱,我们来看看几段代码举例。

1.1、定时器的代码举例:(回调地狱)

 setTimeout(function () {console.log('qiangu1');setTimeout(function () {console.log('qiangu2');setTimeout(function () {console.log('qiangu3');}, 3000);}, 2000);}, 1000);

1.2、Node.js 读取文件的代码举例:(回调地狱)

 fs.readFile(A, 'utf-8', function (err, data) {fs.readFile(B, 'utf-8', function (err, data) {fs.readFile(C, 'utf-8', function (err, data) {fs.readFile(D, 'utf-8', function (err, data) {console.log('qianguyihao:' + data);});});});});

上面代码的逻辑为:先读取 A 文本内容,再根据 A 文本内容读取 B,然后再根据 B 的内容读取 C。为了实现这个业务逻辑,上面的代码就很容易形成回调地狱。

1.3、ajax 请求的代码举例:(回调地狱)

 // 伪代码ajax('a.json', (res1) => {console.log(res1);ajax('b.json', (res2) => {console.log(res2);ajax('c.json', (res3) => {console.log(res3);});});});

2、回调写法不一致的问题

我们需要自己去设计回调函数,包括回调函数的参数格式 、调用方式等等。

 // Node.js 读取文件时,成功回调和失败回调,是通过 error参数来区分readFile('d:\\readme.text', function (err, data) {if (error) {console.log('文件读取失败');} else {console.log('文件读取成功');}});​// jQuery的 ajax 写法中,成功回调和失败回调,是通过两个回调函数来区分$.ajax({url: '/ajax.json',success: function (response) {console.log('文件读取成功');},error: function (err) {console.log('文件读取失败');},});

我们可以看到,上面的回调函数的代码中,成功回调和失败回调,参数的写法不一致。在实战开发中,封装异步函数的人和调用异步函数的人,往往不是同一个人。甚至可能出现的极端的情况是,回调函数里需要传很多参数,参数的顺序也不一致,各有各的风格,每个人写得都不一样。因为这种回调参数的写法不一致、不规范的问题,所以需要单独记忆,导致在调用时需要小心翼翼,很容易出错。

小结

按照上面的分析,在 ES5 中处理异步任务时,产生的这两个问题,ES6 中的 Promise 就可以解决。当然, Promise 的强大功能,不止于此。我们去下一篇文章一探究竟。

赞赏作者

创作不易,你的赞赏和认可,是我更新的最大动力:

希望各位可以点个赞点个关注,这对up真的很重要,谢谢大家啦!

相关文章:

JavaScript异步编程——05-回调函数

我们在前面的文章《JavaScript 基础&#xff1a;异步编程/单线程和异步》中讲过&#xff0c;Javascript 是⼀⻔单线程语⾔。早期我们解决异步场景时&#xff0c;⼤部分情况都是通过回调函数来进⾏。 &#xff08;如果你还不了解单线程和异步的概念&#xff0c;可以先去回顾上一…...

JAVA基础之jsp标准标签

jsp动作标签实现实例化一个实体类 <jsp:useBean id"标识符" class"java类名" scope"作用范围"> 传统的java方式实例化一个实体类 Users user new Users(); <%%> id: 对象名 * class:类 创建对象时,完全限定名(包名…...

VM16激活码以及连接centos7过慢的问题

一、激活码 任选一个&#xff0c;直到能用为止 ZF3R0-FHED2-M80TY-8QYGC-NPKYF YF390-0HF8P-M81RQ-2DXQE-M2UT6 ZF71R-DMX85-08DQY-8YMNC-PPHV8 FA1M0-89YE3-081TQ-AFNX9-NKUC0 二-连接centos7过慢的问题 先备份/etc/ssh/sshd_config,备份命令为 cp /etc/ssh/sshd_config /etc/…...

MySQL 迁移到 Oracle 需要注意的问题

MySQL /Oracle 常见问题 1. VARCHAR/VARCHAR2/NVARCHAR 差异&#xff1a; MySQL 的 VARCHAR 是以字符为单位计算的&#xff0c;Oracle 的 VARCHAR 是 以字节为单位计算的&#xff0c;所以对中文的存储 Oracle 是 MySQL 的 2 倍 (GBK)和 3 倍(UTF8) 2. NULL 差异 A. MySQL…...

【数字经济】上市公司供应链数字化数据(2000-2022)

数据来源&#xff1a; 时间跨度&#xff1a;2000-2022年 数据范围&#xff1a;各上市企业 数据指标&#xff1a; 样例数据&#xff1a; 参考文献&#xff1a;[1]刘海建,胡化广,张树山,等.供应链数字化的绿色创新效应[J].财经研究,2023,49(03):4-18. 下载链接&#xff1a;https:…...

通过AOP实现项目中业务服务降级功能

最近项目中需要增强系统的可靠性&#xff0c;比如某远程服务宕机或者网络抖动引起服务不可用&#xff0c;需要从本地或者其它地方获取业务数据&#xff0c;保证业务的连续稳定性等等。这里简单记录下业务实现&#xff0c;主要我们项目中调用远程接口失败时&#xff0c;需要从本…...

LeetCode:盛最多水的容器

文章收录于LeetCode专栏 盛最多水的容器 给你n个非负整数a1&#xff0c;a2&#xff0c;…&#xff0c;an&#xff0c;每个数代表坐标中的一个点(i, ai) 。在坐标内画 n 条垂直线&#xff0c;垂直线i的两个端点分别为(i, ai) 和 (i, 0)。找出其中的两条线&#xff0c;使得它们与…...

阿里云 OSS桶对象存储攻防

目录 Bucket权限配置错误-公开访问 Bucket桶爆破 特定的Bucket策略配置 Bucket Object遍历...

外网禅道配置

exportfs -avrf 修改代码&#xff0c;避免启动太慢&#xff1a;vi /opt/zbox/bin/zbox.php 启动和停止 /opt/zbox/zbox start /opt/zbox/zbox stop...

MM模块学习一(供应商创建,物料类型的定义及功能)

物料管理流程&#xff1a; 源头&#xff1a;采购需求->采购申请 MRP&#xff1a;物料需求计划。运行物料需求计划的结果&#xff0c;根据物料的性质来判断是外购&#xff08;采购申请&#xff09;或者是生产&#xff08;计划订单->生产订单&#xff09;。 采购申请&am…...

玩comfyui踩过的坑之使用ComfyUI_Custom_NODES_ALEKPET翻译组件问题

环境&#xff1a; 秋叶安装包&#xff0c;安装ComfyUI_Custom_NODES_ALEKPET组件或者直接下载网盘中的包&#xff0c;直接解压包到comfyui根目录/custom_nodes/&#xff0c;重启后&#xff0c;按指导文件操作。 注意&#xff1a;网盘指导包中有配置好的流程json文件&#xff0…...

(类)偏特化Partial Specialization

当编写一个模板特化&#xff0c;涉及部分但不是全部模板参数时&#xff0c;它被称为偏特化&#xff08;Partial Specialization&#xff09;。【注意&#xff0c;偏特化是针对类模板而言&#xff0c;函数模板不可偏特化&#xff0c;只能全特化】 偏特化是C模板编程中的一种技术…...

TypeScript 基础学习笔记:interface 与 type 的异同

&#x1f525; 个人主页&#xff1a;空白诗 文章目录 TypeScript 学习笔记&#xff1a;interface 与 type 的异同&#x1f3a3; 引言&#x1f680; 快速入门1️⃣ Interface&#xff08;接口&#xff09;&#x1f4cb; 定义&#x1f91d; 实现&#x1f4a1; 特点 2️⃣ Type Al…...

【管理咨询宝藏95】SRM采购平台建设内部培训方案

本报告首发于公号“管理咨询宝藏”&#xff0c;如需阅读完整版报告内容&#xff0c;请查阅公号“管理咨询宝藏”。 【管理咨询宝藏95】SRM采购平台建设内部培训方案 【格式】PDF版本 【关键词】SRM采购、制造型企业转型、数字化转型 【核心观点】 - 重点是建设一个适应战略采…...

第七届机电、机器人与自动化国际会议(ICMRA 2024)即将召开!

第七届机电、机器人与自动化国际会议&#xff08;ICMRA 2024&#xff09;将于2024年9月20日-22日在中国武汉举行。ICMRA 2024为各国专家学者提供一个学术交流的平台&#xff0c;讨论机电、机器人和自动化领域的最新研究成果和未来的研究方向&#xff0c;旨在能够建立起国家间&a…...

【智能楼宇秘籍】一网关多协议无缝对接BACnet+OPC+MQTT

在繁华的都市中心&#xff0c;一座崭新的大型商业综合体拔地而起&#xff0c;集购物、餐饮、娱乐、办公于一体&#xff0c;是现代城市生活的缩影。然而&#xff0c;这座综合体的幕后英雄——一套高度集成的楼宇自动化系统&#xff0c;正是依靠多功能协议网关&#xff0c;实现了…...

leetCode68. 文本左右对齐

基本思路&#xff1a; leetCode68. 文本左右对齐 代码 class Solution { public:vector<string> fullJustify(vector<string>& words, int maxWidth) {vector<string> res;for(int i 0; i < words.size(); i){ // 枚举有多少个单词int j i 1; //…...

搜狗输入法 PC端 v14.4.0.9307 去广告绿化版.

软件介绍 搜狗拼音输入法作为众多用户计算机配置的必备工具&#xff0c;其功能的全面性已为众所周知&#xff0c;并且以其高效便捷的输入体验受到广大使用者的青睐。然而&#xff0c;该软件在提供便利的同时&#xff0c;其内置的广告元素常常为用户带来一定的干扰。为此&#…...

【汇总】虚拟机网络不通(Xshell无法连接虚拟机)排查方法

搜索关键字关键字关键字&#xff1a;虚拟机虚拟机虚拟机连接失败、虚拟机无法连接、Xshell连接失败、ping baidu.com失败、静态IP设置 Kali、CentOS、远程连接 描述&#xff1a;物理机无法连接虚拟机&#xff1b;虚拟机无法访问百度&#xff0c;虚拟机无法访问baidu.com 虚拟机…...

C++开发基础之函数参数传递的几种类型

一、前言 在C中&#xff0c;接口指针或类对象的函数参数传递是一个常见的做法&#xff0c;特别是在需要支持多态或动态绑定时。这里将介绍如何传递接口指针或类对象作为函数参数。 二、函数参数传递的几种类型 抽象类&#xff08;接口&#xff09;的实例只能通过指针或引用传…...

使用memcache 和 redis 、 实现session 会话复制和保持

一、NoSQL介绍 NoSQL是对Not Only SQL、非传统关系型数据库的统称 NoSQL一词诞生于1998年&#xff0c;2009年这个词汇再次提出指非关系型、分布式、不提供ACID的数据库设计模式 随着互联网时代的数据爆发时增长、数据库技术发展的日新月异&#xff0c;要适应新的业务需求&am…...

Tomcat 优化

在目前流行的互联网架构中&#xff0c;Tomcat在目前的网络编程中是举足轻重的&#xff0c;由于Tomcat的运行依赖于JVM&#xff0c;从虚拟机的角度把Tomcat的调整分为外部环境调优 JVM 和 Tomcat 自身调优两部分。 一、JVM组成 1. JVM 组成 JVM组成部分 类加载子系统: 使用Ja…...

如何将pdf文件换成3d模型?---模大狮模型网

PDF文件是一种广泛用于文档传输和共享的格式&#xff0c;但在某些情况下&#xff0c;我们可能希望将其中的内容转换为更具交互性和视觉效果的3D模型。本文将介绍如何将PDF文件转换为3D模型&#xff0c;为您展示实现这一想象的步骤。 选择合适的PDF文件&#xff1a; 首先&#…...

Docker 中快速构建 Redis Cluster 集群

Docker 中快速构建 Redis Cluster 集群 目录 前言环境准备 所需软件配置网络 构建 Redis Cluster 镜像 创建自定义 Dockerfile构建镜像 启动 Redis 节点容器 启动命令 配置 Redis Cluster 集群 创建 Redis 集群验证集群状态 总结 前言 Redis 是一个高性能的键值对数据库&am…...

C语言----杨辉三角

各位看官们好。学习到这里想必大家应该对C语言的了解也是很深刻的了吧。但是我们也不能忘记我们一起学习的知识啊。在我们以前学习C语言的时候我想大家应该都听说过杨辉三角吧。虽然我们把其中的规律找到那么这个代码就简单很多了。那么接下里我们就来讲讲杨辉三角。 首先我们先…...

FlaUI

FlaUI是一个基于微软UIAutomation技术&#xff08;简称UIA&#xff09;的.NET库&#xff0c;它主要用于对Windows应用程序&#xff08;如Win32、WinForms、WPF、Store Apps等&#xff09;进行自动化UI测试。FlaUI的前身是TestStack.White&#xff0c;由Roemer开发&#xff0c;旨…...

MySQL调优-01反范式化表设计

MySQL调优-01反范式化表设计 数据库设计三范式 第一范式&#xff08;1NF&#xff09;。确保数据库表的每一列都是不可分割的原子数据项&#xff0c;即列中不可包含数组、记录等非原子数据项&#xff1b;确保表中没有重复的属性或列&#xff0c;每个属性只出现一次&#xff0c;…...

74从零开始学Java之排序算法中的冒泡和选择排序

作者:孙玉昌,昵称【一一哥】,另外【壹壹哥】也是我哦 CSDN博客专家、万粉博主、阿里云专家博主、掘金优质作者 前言 我们要想成为一个优秀的程序员,其实非常关键的一点就是要锻炼培养自己的编程思维,就好比一个狙击手,要通过大量的射击训练要用大量的子弹喂出来。同样的…...

【Qt问题】VS2019 Qt win32项目如何添加x64编译方式

往期回顾&#xff1a; 【Qt问题】Qt常用快捷键汇总-CSDN博客 【Qt问题】Qt Creator 如何链接第三方库-CSDN博客 【Qt问题】Qt 如何带参数启动外部进程-CSDN博客 【Qt问题】VS2019 Qt win32项目如何添加x64编译方式 我们都知道vs2019在编译项目的时候是需要选择编译环境的&…...

LabVIEW换智能仿真三相电能表研制

LabVIEW换智能仿真三相电能表研制 在当前电力工业飞速发展的背景下&#xff0c;确保电能计量的准确性与公正性变得尤为重要。本文提出了一种基于LabVIEW和单片机技术&#xff0c;具有灵活状态切换功能的智能仿真三相电能表&#xff0c;旨在通过技术创新提高电能计量人员的培训…...

东营网站app建设/网络营销的发展现状如何

全国英语等级考试(Public English Test System&#xff0c;简称PETS)是教育部考试中心设计并负责的全国性英语水平考试体系。作为中、英两国政府的教育交流合作项目&#xff0c;在设计过程中它得到了英国专家的技术支持。共有五个级别&#xff1a; PETS-1是初始级&#xff0c;…...

沈阳网站建设q479185700棒/中国疾控卫生应急服装

当我们对数据库优化诊断时&#xff0c;需要收集相应的信息以供参考&#xff0c;从个人的使用经验来说&#xff0c;这种统计数据分为两大类 一类是数据库级别的统计信息 二类是os级别的统计信息 下面就分别介绍在不同的级别下&#xff0c;常用什么工具来收集信息帮助优化诊断 首…...

php网站开发是什么/山西太原网络推广

IoC概念 控制反转&#xff08;Inversion of Control&#xff09;是一个重要的面向对象编程的法则来削减计算机程序的耦合问题。 它还有一个名字叫做依赖注入&#xff08;Dependency Injection&#xff09;。IoC不是什么技术&#xff0c;它是一种设计模式。 实例演示 为了更好的…...

网站建设用哪种语言好/广告公司的业务范围

有些手机用久了&#xff0c;手机的音量就越来越小&#xff0c;相信很多人都碰到过这种情况&#xff0c;这个时候该怎么办呢&#xff1f;有没有办法来提高手机的音量呢&#xff1f;今天我就来介绍3种方法来提升手机的音量。方法一&#xff1a;开启手机的“单声道音频”。很多手机…...

目标网站上做关键字布局/最新经济新闻

正则表达式 正则表达式是指一个用来描述或者匹配一系列符合某个语法规则的字符串的单个字符串。其实就是一种规则。有自己特殊的应用。正则表达式的表达类&#xff1a; 字符类&#xff1a; [abc] a、b 或 c&#xff08;简单类&#xff09;[ ^abc ] 任何字符&#xff0c;除了 a…...

wordpress 网易云音乐插件/爱站工具包的模块有哪些

如何用easyx做图形界面编程创建一个图形窗口(与平时的黑窗口不同哦)贴背景图片循环显示界面游戏控制首先创建一个图形窗口&#xff1a;initgraph(640, 480);//初始化一个大小为640*480的窗口然后让我们给黑窗口加上背景图片&#xff1a;IMAGE img;//注意&#xff1a;要把图片放…...