Promise对象深入理解
目录
基本用法
Promise 对象是一个构造函数,下面的例子生成一个 promise 对象
const promise = new Promise((resolve, reject)=>{
if(true){
resolve('异步操作成功了');
}else{
rejecte('异步操作失败了');
}
})
Promise 构造函数接受一个函数为参数,这个函数还接受两个参数,这两个参数都是函数
第一个函数用于异步成功,并将异步成功的结果以参数的形式传递出去,同时 promise 的状态改为 resolved
第二个函数用于异步失败,并将异步失败的结果以参数的形式传递出去,同时 promise 的状态改为 rejected
返回另一个 Promise 实例
通常情况 reject 函数返回的是异步错误信息,resolve 函数返回的是正常的理想值,而这个值也可以是另一个 Promise 实例,
如果返回的是另一个 Promise 的实例,那么这个实例将决定当前 Promise 实例的状态,自己本来的状态将失效
const p1 = new Promise(function (resolve, reject) {
// ...
}); const p2 = new Promise(function (resolve, reject) {
// ...
setTimeout( () => {resolve(p1),2000 }); // p1 的状态将决定 p2 的状态,而不是等到 2s 后
})
Promise.prototypeof.then
then 方法接受两个函数为参数,通常需要将 Promise 异步的结果作为参数传进去
第一个函数是 异步成功时(resolved)执行
第一个函数是 异步失败时(resolved)执行(可以不传该函数,改用 catch 方法监听异步的失败)
promise.then((res)=>{
console.log(res);
}, (rej)=>{
cnosole.log(rej);
})
then 方法会将新的 Promise 对象,因此可以链式调用。如果有返回值,那么返回值将作为参数传给新的 Promise 对象
promise.then((res)=>{
console.log(res);
return 'newPromise';
}, (rej)=>{
cnosole.log(rej);
}).then((res => {
console.log(res);
})
如果返回的新的 promise对象存在异步操作,例如返回一个 新的 Promise 实例对象,那么后面的 then 方法会等到这个promise状态发生改变时才执行
Promise.prototype.catch
上面有说到 then 方法接收两个函数为参数,分别是成功和失败时执行的,并且建议失败的函数改用 catch 方法
catch 可以实现 then 方法第二个参数的功能,但是它可以上代码跟直观
let a = 1;
let promise = new Promise(function(resolve, reject){
if(a==10){
resolve('成功');
}else{
reject('失败');
}
}); promise.then(res=>{
console.log(res);
}).catch(err=>{
console.log(err);
})
promise 对象的错误具有“冒泡”性质,会一直向后传递,直到被捕获为止。也就是说,错误总是会被下一个catch
语句捕获。
Promise.prototype.finally()
finally 方法用于指定 不管 Promise 对象最后状态如何,都会执行的操作。
promise
.finally(() => {});
Promise.resolve()
将现有对象转换为 promise 对象,分下面四种情况
1. 如果参数是 Promise 实例,那么 promise.resolve 将不做任何修改、原封不动地返回这个实例。
2. 如果参数是一个 thenable 对象。thenable 对象指的是具有 then 方法的对象,比如下面这个对象。Promise.resolve 方法会将这个对象转为 Promise 对象,然后就立即执行thenable
对象的then
方法。
let thenable = {
then: function(resolve, reject) {
resolve(42);
}
}; let p1 = Promise.resolve(thenable);
p1.then(function(value) {
console.log(value); //
});
3. 如果参数是一个原始值,或者是一个不具有then
方法的对象,则 Promise.resolve 方法返回一个新的 Promise 对象,状态为 resolved
。下面代码生成一个新的 Promise 对象的实例p
。由于字符串 Hello
不属于异步操作(判断方法是字符串对象不具有 then 方法),返回 Promise 实例的状态从一生成就是 resolved
,所以回调函数会立即执行。Promise.resolve
方法的参数,会同时传给回调函数。
const p = Promise.resolve('Hello'); p.then(function (s){
console.log(s)
});
// Hello
4. Promise.resolve 方法允许调用时不带参数,直接返回一个 resolved 状态的 Promise 对象。需要注意的是,立即 resolve() 的 Promise 对象,是在本轮“事件循环”(event loop)的结束时执行,而不是在下一轮“事件循环”的开始时。
setTimeout(function () {
console.log('three');
}, 0); // 下一轮“事件循环”开始时执行, Promise.resolve().then(function () {
console.log('two');
}); // 在本轮“事件循环”结束时执行 console.log('one'); // one
// two
// three
Promsie.reject()
同样也是将现有对象转换为 promise 对象,只不过 promise 对象的状态是 rejected
const p = Promise.reject('出错了');
Promise 的执行顺序
Promise 构造函数本身是同步执行的
then 方法会被放到微任务(microtask)队列中
setTimeout 定时器会放到宏任务(macrotask)队列中
当同步任务执行完后,微任务队列中的任务依次进入主线程执行,当微任务队列为空以后,宏任务队列中的任务依次进入主线执行
setTimeout(() => {
console.log(0);
},0) new Promise(resolve => {
resolve(1);
Promise.resolve().then(t => {
console.log(2);
})
console.log(3);
}).then(t => {
console.log(t);
}) console.log(4);
//
//
//
//
//
现在分析上面的代码
1. 遇到定时器,将它放到宏任务队列中(这是第一个宏任务)
2. 同步执行构造函数
3. 在构造函数中遇到一个 then 方法,将他放到微任务队列中(这是第一个微任务)
4. console.log(3) 执行,打印出 3
5. 遇到第二个 then 方法,将他放到微任务队列的末尾
6. console.log(4) 执行,打印出 4
7. 读取微任务队列中的任务,依次打印出 2 1
8. 读取宏任务队列中的任务,打印出 0
Promise对象深入理解的更多相关文章
- ES6中Promise对象个人理解
Promise是ES6原生提供的一个用来传递异步消息的对象.它减少了传统ajax金字塔回调,可以将异步操作以同步操作的流程表达出来使得代码维护和可读性方面好很多. Promise的状态: 既然是用来传 ...
- 谈谈 ES6 的 Promise 对象
https://segmentfault.com/a/1190000002928371 前言 开篇首先设想一个日常开发常常会遇到的需求:在多个接口异步请求数据,然后利用这些数据来进行一系列的操作.一般 ...
- ES6的promise对象研究
ES6的promise对象研究 什么叫promise? Promise对象可以理解为一次执行的异步操作,使用promise对象之后可以使用一种链式调用的方式来组织代码:让代码更加的直观. 那我们为什么 ...
- angularJS中的Promise对象($q)的深入理解
原文链接:a better way to learn AngularJS - promises AngularJS通过内置的$q服务提供Promise编程模式.通过将异步函数注册到promise对象, ...
- 彻底理解Promise对象——用es5语法实现一个自己的Promise(上篇)
本文同步自我的个人博客: http://mly-zju.github.io/ 众所周知javascript语言的一大特色就是异步,这既是它的优点,同时在某些情况下也带来了一些的问题.最大的问题之一,就 ...
- 大白话理解promise对象
Promise 代表了未来某个将要发生的事件(通常是一个异步操作) Promise 是异步编程的解决方案,能够简化多层回调嵌套,代表了未来某个将要发生的事件.Promise是一个构造函数,本身有a ...
- ES6的promise对象应该这样用
ES6修补了一位Js修真者诸多的遗憾. 曾几何时,我这个小白从js非阻塞特性的坑中爬出来,当我经历了一些回调丑陋的写法和优化的尝试之后,我深深觉得js对于多线程阻塞式的开发语言而言,可能有着其太明显的 ...
- JavaScript异步编程(1)- ECMAScript 6的Promise对象
JavaScript的Callback机制深入人心.而ECMAScript的世界同样充斥的各种异步操作(异步IO.setTimeout等).异步和Callback的搭载很容易就衍生"回调金字 ...
- Promise对象
1.Promise思想:每一个异步任务立刻返回一个Promise对象,由于是立刻返回,所以可以采用同步操作的流程.这个Promises对象有一个then方法,允许指定回调函数,在异步任务完成后调用. ...
随机推荐
- python的self
python类定义里面的self就是指的该类的对象本身.
- Thumbelina,摘自iOS应用Snow White and more stories
Once upon a time there was a woman who wanted to have a child. 从前,有个想要个孩子的女人. A witch heard her wish ...
- Android Weekly Notes Issue #246
Android Weekly Issue #246 February 26th, 2017 Android Weekly Issue #246 本期内容包括: RecyclerView上的Shared ...
- Ext rowcontextmenu回调
今天栽了大跟头,,,, EXT js右键可以获取到选中的那一列的索引值,只需在回调函数中取到那一列的数据就可以ok,不用再费尽心思的费用执行click的事件 grid.on("rowcont ...
- 科目三靠边停车难度升级,超过50cm不合格怎么破?
驾考新规实施了几天,考过的学员普遍反映科目三难度升级,其中靠边停车项目的扣分点细化之后,一不小心就会不合格.新规以前靠边停车不压线就好了,新规之后,车辆距离马路右侧边缘线超过50cm就要扣100分,对 ...
- The import ....cannot be resolved 解决方法
1:右击项目build path>configure build path>libraries看有没感叹号什么的不正常的lib,移除掉 2:点击项目的build path>confi ...
- ModuleNotFoundError: No module named 'numpy.core._multiarray_umath' ImportError: numpy.core.multiarray failed to import
出现以下错误:可能是因为你的numpy版本太低 更新numpy的版本 pip install -i https://pypi.tuna.tsinghua.edu.cn/simple --upgra ...
- iOS-个人开发者账号转公司开发者账号(邓白氏码申请教程)
邓白氏编码申请 个人开发者账号转公司开发者账号,首先要申请邓白氏编码-DUNS,打开https://developer.apple.com/support/进行DUNS申请! 步骤如下: 1.选择Me ...
- ora-12519:TNS ,no appropriate service handler found
今天有同事反应,数据库连不了,提示 ora-12519:TNS ,no appropriate service handler found: 一开始以为监听没有启动,排查后,发现正常:于是google ...
- CISCO-从TFTP上上传/下载配置文件
1.下载配置文件到TFTP服务器: 2.上传配置文件到路由器