Node.js异步编程之Promise
发表于|更新于
|字数总计:1.1k|阅读时长:4分钟|阅读量:
前言
先来介绍一下Promise是什么?
Promise:
- 当前事件循环得不到的结果,但未来的事件循环会给到你结果
- 是一个状态机
- pending——还没有得到结果
- fulfilled/resolved——得到了一个正确的结果
- rejected——得到了一个错误的结果
从代码入手👻
1 2 3 4 5 6 7 8 9 10 11 12 13
| (function(){ var promise = new Promise(function(resolve,reject){ setTimeout(()=>{ resolve(); },500) }) console.log(promise); setTimeout(()=>{ console.log(promise); },800) })();
|
这是一个从pending到resolved/rejected的过程。
那么可能有的小伙伴会想,我可不可以让他在300毫秒的时候转变成resolve的状态,在500毫秒的时候从resolved转变成rejected状态呢?那我们来实际操作一下看看结果吧。
我也有这样的疑惑
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| (function(){ var promise = new Promise(function(resolve,reject){ setTimeout(()=>{ resolve(); },300) setTimeout(()=>{ reject(new Error()); },300) }) console.log(promise); setTimeout(()=>{ console.log(promise); },800) })();
|
从结果我们可以看到,他还是fulfilled状态,也就是resolved状态,没有改变。
所以在resolved状态与rejected状态之间应该是不能转换的。
继续学习🐸
我们使用promise去解决异步问题,是希望他拿到结果之后就要立即通知我们,那这个时候我们要怎么去写呢?
我们可以借助promise下的一个方法,叫做then
把上面的代码进行改写
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| (function(){ var promise = new Promise(function(resolve,reject){ setTimeout(()=>{ resolve(3); },300) }).then(function(res){ console.log(res) }).catch(function(){ }) console.log(promise); setTimeout(()=>{ console.log(promise); },800) })();
|
可能有小伙伴注意到了,我们上面还多了一个catch方法,没错,他在这里也是用来处理error的,而error在promise里就对应着rejected状态了,将上面的代码再改点东西,就成了这样子。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| (function(){ var promise = new Promise(function(resolve,reject){ setTimeout(()=>{ reject(new Error()); },300) }).then(function(res){ console.log(res) }).catch(function(err){ console.log(err) }) console.log(promise); setTimeout(()=>{ console.log(promise); },800) })();
|
- 关于.then和.catch
- resolved状态的promise会回调后面的第一个.then
- rejected状态的promise会回调后面的第一个.catch
- 任何一个rejected状态且后面没有.catch的promise,都会造成浏览器/node环境的全局错误
为什么说promise优秀呢?
他可以解决异步流程控制的问题(这个在前面callback那节笔记有记过,[可以去看一下](Node.js学习笔记之异步编程部分(一) | Wangez-Blog))。
还是之前面试的问题,这次我们用promise的思路来看一下
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
| (function(){ var promise = interview(); promise .then((res)=>{ console.log('smile'); }) .catch((err)=>{ console.log('cry') })
})();
function interview(){ return new Promise((resolve,reject)=>{ setTimeout(()=>{ if(Math.random()>0.2){ resolve('success') }else{ reject(new Error('fail')); } },500) }) }
|
小细节,拿笔记好📢
执行then和catch会返回一个新的promise,该promise最终状态根据then和catch的回调函数的执行结果决定
- 如果回调函数是throw,那么promise就是rejected状态
- 如果回调函数是return,那么promise就是resolved状态
- 但如果回调函数最终return了一个promise,该promise回跟回调函数return的那个promise状态保持一致。
接下来用多轮面试的例子,来演示一下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32
| (function(){ var promise = interview(); promise .then((res)=>{ return interview(2) }) .then(()=>{ return interview(3) }) .then(()=>{ console.log('smile'); }) .catch((err)=>{ console.log('cry at ' + err.round + ' round'); })
})();
function interview(round){ return new Promise((resolve,reject)=>{ setTimeout(()=>{ if(Math.random()>0.2){ resolve('success') }else{ var error = new Error('fail'); error.round = round reject(error); } },500) }) }
|
想知道你在第几轮面试挂了吗?去执行一下试试吧!
小优化🎁
我们肯定是同时面试多家公司的呀,那我们再给上面的代码做一些优化吧。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
| (function(){ Promise .all([ interview('tencent'), interview('Ali88') ]) .then(()=>{ console.log('smile'); }) .catch((err)=>{ console.log('cry for ' + err.name ); })
})();
function interview(name){ return new Promise((resolve,reject)=>{ setTimeout(()=>{ if(Math.random()>0.2){ resolve('success') }else{ var error = new Error('fail'); error.name = name reject(error); } },500) }) }
|
注意:
- catch只能返回第一个面试挂的公司,如果都想知道就要存好所有的promise的状态并且打印出来哦。
最后
肝不动了,睡了睡了,总觉得上面那个例子,面试通过一家,第二家挂了他会给我返回到catch里面去呢,明天在研究一下。