在谈及具体的异步编程前,我们还需要知道何为同步编程,何为异步编程,这两者工作可是有着天差地别。
那让我来举个栗子吧:假如你需要去做两件事,分别是去煮饭,听一首歌,那么,煮完饭再去听歌的执行流程是同步编程流程。而先煮饭,在煮饭的过程中去听歌的执行流程则是异步编程流程。
JS原本的任务就是负责浏览器和用户之间的交互功能,因为JS要操作DOM,所以必须是单线程的,如果你是JS的话,请你想想如果有多个线程同时操作同一个DOM,想必你会抓狂的吧。正因为如此,所以排队等待时间未免太过漫长,所以才有了我们现在口中的JS异步编程的方案。
实现异步编程的方案有很多,例如:回调函数、promise对象,事件监听、发布/订阅等等方案,但万变不离其中,所有异步编程的核心就是回调函数。下面我来举例具体说明一下什么是回调函数:
假如在煮好饭后要把听歌的耳机放到耳机架上,那么就要把这个事告诉你对象,那异步流程就变成了——你先煮饭,然后告诉你对象等你听完歌需要把耳机放到耳机架上,然后你再去听歌。在这个流程中,把耳机放回耳机架是回调函数,也就是在任务完成后你需要执行的操作,你就是异步操作的调用者,你对象就是异步操作的执行者,总结起来就是——由调用者定义,交给执行者执行的函数,就是回调函数。
说起回调地狱,想必很多小伙伴并不陌生,这个问题也困扰着当时初学JS的我,想当初可是被这个问题折磨了一段时间。接下来就由我给大家说说回调地狱,具体来说回调地狱就是,当你执行DOM事件时,比如在页面上点击链接、回车等操作,浏览器都会悄悄地向服务端发送很多个http请求,携带后台可识别的参数,等待服务器返回数据,这个过程是异步回调的,当很多功能环环依赖时,代码会一层一层的嵌套起来,看起来十分庞大且恶心.
正是因为回调地狱不利于阅读,同时也不利于维护,所以前辈们为了解决这一问题,在很多方面做出了努力,这个时候我要跟大家分享一个JS异步编程的一个核心要点——promise,下面请用心分析我所说的内容:
(一)何为promise?
promise是一个构造函数,返回一个promise对象。一个promise指的是一个可能会在未来的某个时间点产生一个单一值的对象。我这里有一个很好的说法可以描述一下promise的机制:
好比你去汉堡店买汉堡包吃,你已经付好了钱,但是却被告知汉堡包已经售空,而店员为了解决你的问题,给了你一张汉堡兑换券,这张券不会过期,她让你明天再来拿券兑换汉堡包。虽然你的券可以换到汉堡包,但是你也不能确定你明天来时是否有汉堡可以兑换,但可以确定的是,你一定能在未来的某一天,兑换到这个汉堡包。
(二)promise自身的三种状态 (状态改变后就不会再变化)
(1)fulfilled:成功
(2)rejected:失败
(3)pending:等待
(三)promise的用法
promise给定了一些原型方法来进行状态处理
1.promise.prototype.then :
promise对象通过内部resolve函数完成状态的pending ->fulfilled。之后promise将保留这个状态。可以通过then去处理状态。例:pm.then( (val) =>{console.log(val); } )
处理rejected状态,then方法可接收两个函数
new Promise(function(resolve,reject){
setTimeout(() =>{
reject(new Error('err'))
},2000).then(null,(error) =>{
console.log(error);
})
})
2.promise.prototype.catch:
用来做错误统一处理,这样的链式调用中我们只需要用then来处理fulfilled状态,在链的末尾加上catch来统一处理错误。
new Promise((resolve,reject) =>{
setTimeout(() =>{
resolve = 100;
},2000).then(result =>{
return result * num;
return result / 2;}).catch(err =>{
console.log(err);
})
})
3.promise.prototype.finally():
这个API放在结尾,这时候你们肯定会好奇这是为什么呢?因为它不管前面的promise变为什么,它都会执行里面的回调函数。
new Promise((resolve,reject) =>{
setTimeout(() =>{
resolve = 100;
},2000).then(result =>{
return result * num;
return result / 2;}).catch(err =>{
console.log(err);
console.log('complete')
})
})
4.promise.all
Promise.all([A,B]).then(resA,resB) =>{
//do something
}
5.promise链式调用
promise对象的then方法会返回一个权限的promise对象
后面的then方法就是为了上一个then返回的promise注册回调
前面的then方法中回调函数的返回值会作为后面then方法回调的参数
如果回调返回的是Promise,后面的then方法的回调会等待它的结束
EventLoop被称作事件轮询,也称为事件循环,而EventLoop的作用只有一个:监听调用栈和消息队列,一旦调用栈清空,EventLoop就会从消息队列中取出第一个回调函数,压入栈中执行它。
而就我个人的观点来看,我是这么理解它的:代码从上到下执行,如果遇到微任务就放到微任务队列中,遇到宏任务就放到宏任务队列中,所有代码被执行栈执行完后,先去微任务列表中执行任务,待解决所有微任务后,再去宏任务队列解决所有宏任务。
注意:
1.每当执行完当前宏任务时,都会返回微任务列表看是否有微任务需要执行,若存在微任务则先执行微任务,若没有微任务存在,则执行剩下的宏任务。
2.微任务不需要到队列队尾进行排队处理。
今天的分享就到这啦,这里是一条从艺术行业初入前端的小咸鱼,欢迎大家提出宝贵的建议,并一起讨论前端的技术知识,希望大家都能在不久的将来成为一名前端老鸟。