-
定时器的回调
-
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 成功的解决了回调地狱的问题,它又不是异步编程的终极方案,那它又带来了什么问题呢?
-
无法取消 Promise
-
当处于 pending 状态时是,无法得知进展
-
错误不能被
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
使用了
异步迭代生成器
小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级前端工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Web前端开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频
如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注:前端)
基础学习:
前端最基础的就是 HTML , CSS 和 JavaScript 。
网页设计:HTML和CSS基础知识的学习
HTML是网页内容的载体。内容就是网页制作者放在页面上想要让用户浏览的信息,可以包含文字、图片、视频等。
开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】
CSS样式是表现。就像网页的外衣。比如,标题字体、颜色变化,或为标题加入背景图片、边框等。所有这些用来改变内容外观的东西称之为表现。
动态交互:JavaScript基础的学习
JavaScript是用来实现网页上的特效效果。如:鼠标滑过弹出下拉菜单。或鼠标滑过表格的背景颜色改变。还有焦点新闻(新闻图片)的轮换。可以这么理解,有动画的,有交互的一般都是用JavaScript来实现的。
结学习笔记+真实项目实战+最新讲解视频】](https://bbs.csdn.net/topics/618166371)**
[外链图片转存中…(img-bIvTJxOp-1710772321246)]
CSS样式是表现。就像网页的外衣。比如,标题字体、颜色变化,或为标题加入背景图片、边框等。所有这些用来改变内容外观的东西称之为表现。
[外链图片转存中…(img-3VFtw7eu-1710772321247)]
动态交互:JavaScript基础的学习
JavaScript是用来实现网页上的特效效果。如:鼠标滑过弹出下拉菜单。或鼠标滑过表格的背景颜色改变。还有焦点新闻(新闻图片)的轮换。可以这么理解,有动画的,有交互的一般都是用JavaScript来实现的。