ES6:【深扒】深入理解 JavaScript 中的异步编程

  1. 回调函数

  2. 事件监听

  3. 发布/订阅

  4. Promise 对象

下面来先来回顾以下传统方法是如何实现异步编程的

Callback

回调函数可以理解为一件想要去做的事情,由调用者定义好函数,交给执行者在某个时机去执行,把需要执行的操作放在函数里,将函数传入给执行者执行

主要体现在,把任务的第二段写在一个函数里面,等到重新执行这个任务的时候,直接调用

那有人就会问了,第二段是指什么,我们再举一个例子,读取文件进行打印,这个操作肯定是异步的吧,那它怎么分两段呢?

按照逻辑来分,第一段是读取文件,第二段是打印文件,可以理解为第一段是请求数据,第二段是打印数据

阮老师的代码实例

fs.readFile(‘/etc/passwd’, ‘utf-8’, function (err, data) {

if (err) throw err;

console.log(data);

});

在第一阶段执行结束后,会将结果返回给后面的函数作为参数,传入第二段

回调函数的使用场景:

  1. 事件回调

  2. 定时器的回调

  3. Ajax 请求

Promise

采用回调函数的方法,本身是没有问题的,但是问题出现在多个回调函数的嵌套

想一想,我执行完执行你,你执行完执行他,他执行完又执行她…

是不是需要层层嵌套,那这样套娃式的操作显然不利于阅读

fs.readFile(fileA, ‘utf-8’, function (err, data) {

fs.readFile(fileB, ‘utf-8’, function (err, data) {

// …

});

});

同时你也可以这样去思考一下,如果有其中一个代码需要修改,那它的上层回调和下层回调都要修改,这也叫做强耦合

耦合,藕断丝连,关联性很强的意思

这种场景也叫做“回调地狱”

而 Promise 对象的诞生就是为了解决这个问题,它采用了以一种全新的写法,链式调用

Promise 可以用来表示一个异步任务执行的状态,有三种状态

  • Pending:开始是等待状态

  • Fulfilled:成功的状态,会触发 onFulfilled

  • Rejected:失败的状态,会触发 onRejected

它的写法如下

const promise = new Promise(function(resolve, reject) {

// 同步代码

// resolve执行表示异步任务成功

// reject执行表示异步任务失败

resolve(100)

// reject(new Error(‘reject’)) // 失败

})

promise.then(function() {

// 成功的回调

}, function () {

// 失败的回调

})

Promise 对象调用 then 方法后会返回一个新的 Promise 对象,这个新的 Promise 对象可以继续调用 then 实现链式调用

后面的 then 方法是为上一个 then 返回的 Promise 对象注册回调

前一个 then 方法中回调函数的返回值会作为后面 then 方法回调的参数

链式调用的目的是为了解决回调函数嵌套的问题

关于 Promise 的更多细节这里就不多说了,下一篇写吧~

坏了,坏了,环环嵌套,我陷入回调地狱了,努力更文

Promise 成功的解决了回调地狱的问题,它又不是异步编程的终极方案,那它又带来了什么问题呢?

  1. 无法取消 Promise

  2. 当处于 pending 状态时是,无法得知进展

  3. 错误不能被 catch

但是这些都不是 Promise 的最大问题,它最大的问题是代码冗余,当执行逻辑变得复杂时,代码的语义会变得很不清楚,全是 then

其实看过上一篇文章的读者们,看到这里应该对 Generator 实现异步编程有了一定的眉目,这里的 then 方法的作用,似乎 next 方法也能实现,启动,运行,传参,接下来我们来细说一下

Generator

Generator 函数可以暂停执行恢复执行, 这是它能封装异步任务的根本原因。

除此之外,它还有两个特征,使它可以作为异步编程的完美解决方案。

  • 函数体内外的数据传递

  • 错误处理机制

数据传递

在学习它是如何实现异步编程的之前,我们先回顾一下 Generator 函数的执行方法

// 声明Generator函数

function* gen(x){

let y = yield x + 2

return y

}

// 遍历器对象

let g = gen()

// 第一次调用next方法

g.next() // { value: 3, done: false }

// 第二次调用 传递参数

g.next(2) // { value: 2, done: true }

首先执行 gen 函数,获得遍历器对象,此时函数并不会执行,当调用遍历器对象的 next 方法时,执行到第一个 yield 语句,以此类推

也就是说只有调用 next 方法,才会往下执行

同时在上面的代码中,我们可以通过 value 来获取返回的值,通过给 next 方法传递参数来实现数据交换

错误处理机制

Generator 函数内部可以部署错误处理代码,捕获函数体外抛出的错误

function* gen(x){

try {

var y = yield x + 2;

} catch (e){

console.log(e);

}

return y;

}

var g = gen(1);

g.next();

g.throw(‘出错了’);

或许会有人不理解为什么内部的 catch 可以捕获外部的错误?

原因是我们通过 g.throw 来抛错误,其实是将错误抛入了生成器,毕竟我们是在 p 上来调用 throw 方法

实现异步编程

在我的上一篇文章详细的介绍了生成器的执行机制,以及 yield 执行特点,可以先阅读一下

我们主意利用 yield 暂停生成器函数执行的特点,来使用生成器函数去实现异步编程,我们来看一个例子

Generator + Promise

function * main () {

const user = yield ajax(‘/api/usrs.json’)

console.log(user)

}

const g = main()

const result = g.next()

result.value.then(data => {

g.next(data)

})

首先我们定义一个生成器函数 main ,然后在这个函数内部使用 yield 去返回一个 ajax 的调用,也就是返回了一个 Promise 对象。

然后去接收 yield 语句的返回值,也就是第二个 next 方法的参数。

我们可以在外界去调用生成器函数得到它的迭代器对象,然后调用这个对象的 next 方法,这样 main 函数就会执行到第一个 yield 的位置,也就是会执行到 ajax 的调用,这里 next 方法返回对象的 value 值就是 ajax 返回的 Promise 对象

因此我们可以通过 then 方法去指定这个 Promise 的回调,在这个 Promise 回调中我们就可以拿到这个 Promise 的执行结果 data,这时候我们就可以通过再调用一次 next 方法,把我们得到的 data 数据传递出去,这样 main 函数就可以继续执行了,而 data 就会被当作 yield 表达式的返回值赋值给 user 使用了

异步迭代生成器

如果上面的 generator + promise 能够理解的话,这个就更简单了,就是单纯的使用 generator 实现的异步编程

function foo(x, y) {

ajax(“1.2.34.2”, function(err,data) {

if(err) {

it.throw(err)

}else {

it.next(data)

}

})

}

function *main() {

let text = yield foo(11, 31)
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数前端工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Web前端开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上前端开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:前端)

最后

文章到这里就结束了,如果觉得对你有帮助可以点个赞哦,如果有需要前端校招面试题PDF完整版的朋友可以点击这里即可免费获取,包括答案解析。

)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上前端开发知识点,真正体系化!

[外链图片转存中…(img-BDOu57zE-1713485451909)]

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:前端)

[外链图片转存中…(img-3S8jX3NH-1713485451909)]

最后

文章到这里就结束了,如果觉得对你有帮助可以点个赞哦,如果有需要前端校招面试题PDF完整版的朋友可以点击这里即可免费获取,包括答案解析。

[外链图片转存中…(img-iNLRTBTT-1713485451910)]

  • 11
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值