当前位置:网站首页>es6新增-Promise详解(异步编程的解决方案1)
es6新增-Promise详解(异步编程的解决方案1)
2022-08-03 17:38:00 【卷心菜007】
目录
2.2.一旦状态改变(确定),就不会再变了,Promise对象的状态改变,只有两种可能:
3.promise实例生成以后,可以用then方法分别指定resolved状态和rejected状态的回调函数
一. Promise诞生背景
1.1作用
异步编程的一种解决方式,用来解决回调地狱(回调套回调)。
1.2场景
所有的异步回调代码(定时器/ajax请求)都可以用Promise
1.3回调地狱示例
开始=>2s 买菜=>2s 洗菜=>2s 切菜=>2s 炒菜=>结束
console.log("开始")
setTimeout(()=>{console.log("买菜")
setTimeout(()=>{console.log("洗菜")
setTimeout(()=>{console.log("切菜")
setTimeout(()=>{console.log("炒菜")
console.log("结束")
},2000)
},2000)
},2000)
},2000)
二. Promise实例的状态 (!important)
2.1.每个Promise实例有三种状态,分别是
pending(进行中),fulfilled(已成功),rejected(已失败)
创建时候的默认状态是pending,只有异步操作的结果,可以决定当前是哪一种状态,任何操作都无法改变这个状态。这也是Promise名字的由来。
2.2.一旦状态改变(确定),就不会再变了,Promise对象的状态改变,只有两种可能:
a.从pending变为fulfilled b.从penging变为rejected
三. Promise的使用
1.Promise的基本使用
Promise是es6的新增的一个构造函数,用来生成Promise实例
const p = new Promise((resolve,reject)=>{
//...异步代码(也可以放同步代码)
//如:ajax请求、定时器、事件、读文件、操作数据库等
if( 异步操作成功 ){ resolve(value) }
else{ reject(error) }
})
2.resolve和reject参数的作用
Promise构造函数接受一个函数作为参数,该函数的两个参数分别为resolve和reject.参数的作用:
resolve的作用:
将Promise对象的状态从"未完成"变为"已成功",即pending=>resolved
reject的作用:
将Promise对象的状态从"未完成"变为"已失败",即pending=>rejected,
在异步操作失败时调用 , 并将异步操作报出的错误 , 作为参数传递出去
3.promise实例生成以后,可以用then方法分别指定resolved状态和rejected状态的回调函数
p.then((value)=>{
//成功 } , (error){ //失败 })
then方法可以接受两个回调函数作为参数,
第一个回调函数是Promise对象的状态变为resolved时调用, 成功的回调
第二个回调函数是Promise对象的状态变为rejected时调用,失败的回调
这两个函数都是可选的,不一定都要提供,他们接受Promise对象传出的值作为参数
4.Promise解决回调地狱示例
(凡是异步的操作,都可以用Promise的方式来改造)
function cook(stage){
return new Promise((resolve)=>{
setTimeout(()=>{ resolve(stage) },2000) })
}
console.log("开始");
cook("买菜").then((value)=>{
console.log(value) })
控制台打印: 开始=>2s 买菜
5.进一步认识Promise构造函数
5.1 const p = new Promise( (resolve,reject)=>{
setTimeout( ()=>{
//Promise的状态一旦确定,就不可以再变更了
reject("穷,一个人凉快去吧")
},2*1000 )
} )
p.then((value)=>{ console.log(value)
},error=>{ console.log(error) } )
两秒后输出:穷,一个人凉快去吧,执行第二个回调函数
5.2 const p = new Promise( (resolve,reject)=>{
setTimeout( ()=>{
//Promise的状态一旦确定,就不可以再变更了
resolve("富有,迎娶白富美")
reject("穷,一个人凉快去吧")
//在resolve执行之后,Promise状态变为成功,就不再改变了
},2*1000 )
} )
p.then((value)=>{ console.log(value)
},error=>{ console.log(error) } )
两秒后输出:富有,迎娶白富美,执行第一个回调
5.3 function cook(stage){
return new Promise((resolve)=>{
setTimeout(()=>{ reject("钱不够,没买到") },2000) })
}
console.log("开始");
// .then 表示成功的回调,添加error表示失败的回调
cook("买菜").then((value)=>{
console.log(value) },(error)=>{ console.log(error) } )
控制台打印:开始=>2s 钱不够,没买到
注:成功回调走.then,失败回调走error
四. Promise的实例方法和静态需求
1.Promise的实例方法then
1.1 then的介绍
Promise实例具有then方法,也就是说,then方法是定义在原型对象Promise、prototype上的。
作用:为Promise实例添加改变状态时的回调函数
参数:then方法的第一个参数是resolved状态(成功态)的回调函数,第二个参数是rejected状态(失败态)的回调函数,他们都是可选的。
1.2 then的返回值
then方法返回的是一个新的Promise实例
(注意:不是原来那个Promise实例,状态又从pending开始)
因此,可以采用链式写法,即then方法后面再调用另一个then方法。
const p = new Promise( ( resolve,reject )=>{
setTimeout( ()=>{
resolve("成功")
//reject("失败")
},1000 )} )
p
.then((v1)=>{
console.log(v1)
//1.返回一个普通值(表明新的Promise实例状态是成功的)
return 1
//2.返回一个新的Promise实例,状态由resolve或reject函数决定
return new Promise((resolve,reject)=>{
setTimeout( ()=>{
resolve("success")
//reject("fail")
} )
},1000)
},error=>{
console.log(error)
return 2
} )
.then((v2)=>{ console.log("v2") },error=>{ console.log(error) } )
不同情况输出:
const p = new Promise( ( resolve,reject )=>{
setTimeout( ()=>{
resolve("成功")
},1000 )} )
p
.then((v1)=>{
console.log(v1)
return 1
},error=>{
console.log(error)
return 2
} )
.then((v2)=>{ console.log("v2") },error=>{ console.log(error) } )
输出:成功 1
const p = new Promise( ( resolve,reject )=>{
setTimeout( ()=>{
resolve("成功")
},1000 )} )
p
.then((v1)=>{
return new Promise((resolve,reject)=>{
setTimeout( ()=>{
resolve("success")
} )
},1000)
},error=>{
console.log(error)
return 2
} )
.then((v2)=>{ console.log("v2") },error=>{ console.log(error) } )
输出:成功 success
const p = new Promise( ( resolve,reject )=>{
setTimeout( ()=>{
reject("失败")
},1000 )} )
p
.then((v1)=>{
return new Promise((resolve,reject)=>{
setTimeout( ()=>{
resolve("success")
} )
},1000)
},error=>{
console.log(error)
return 2
} )
.then((v2)=>{ console.log("v2") },error=>{ console.log(error) } )
输出:失败 success
const p = new Promise( ( resolve,reject )=>{
setTimeout( ()=>{
reject("失败")
},1000 )} )
p
.then((v1)=>{
return new Promise((resolve,reject)=>{
setTimeout( ()=>{
reject("fail")
} )
},1000)
},error=>{
console.log(error)
return 2
} )
.then((v2)=>{ console.log("v2") },error=>{ console.log(error) } )
输出:失败 fail
注:下一个.then走成功还是失败,取决于前一个.then的回调中返回的是resolve还是reject
1.3 then链式结构解决多重嵌套回调地狱问题
function cook(stage){
return new Promise((resolve)=>{
setTimeout(()=>{ resolve(stage) },2000) })
}
console.log("开始");
cook("买菜").then(v1=>{
console.log(v1)
return cook("洗菜")
})
.then(v2=>{
console.log(v2)
return cook("切菜")
})
.then(v2=>{
console.log(v3)
return cook("炒菜")
})
.then(v4=>{
console.log(v4)
console.log("结束")
})
控制台打印: 开始=>2s 买菜=>2s 洗菜=>2s 切菜=>2s 炒菜=>结束
2.Promise的实例方法catch
Promise.prototype.catch()方法是
.then(null,rejectCallback) 或.then(undefined,rejectCallback)的别名
用于指定Promise发生错误/失败时的回调函数
const p = new Promise((resolve,reject)=>{
setTimeout(()=>{
reject("出错啦")
},1000)
})
//如果 只是处理失败有两种方法
//1.then 的第二个回调
p
.then(null,error=>{
console.log(error)
})
//2.等价于catch方法
p
.catch(error=>{
console.log(error)
})
输出:出错啦
3.Promise的实例方法finally
Promise.prototype.finally()
finally()方法的作用:不管Promise实例最后状态是成功还是失败,都会执行的操作,该方法是es2018引入标准的
类似于代码块:
try{
}catch(e){
}finally{
}
示例
const p = new Promise((resolve,reject)=>{
setTimeout(()=>{
resolve("成功");
//reject("失败")
},1000)
})
p
.then(value=>{
console.log(value)
})
//catch来捕获回调的失败状态,等同于.then的error
.catch(error=>{
console.log(error)
})
.finally(()=>{
console.log("finally无论状态成功还是失败,都会执行")
})
输出:成功 finally无论状态成功还是失败,都会执行
const p = new Promise((resolve,reject)=>{
setTimeout(()=>{
reject("失败")
},1000)
})
p
.then(value=>{
console.log(value)
})
//catch来捕获回调的失败状态,等同于.then的error
.catch(error=>{
console.log(error)
})
.finally(()=>{
console.log("finally无论状态成功还是失败,都会执行")
})
输出:失败 finally无论状态成功还是失败,都会执行
4.Promise的静态方法
Promise.resolve()
将现有的对象转换为 一个成功态的Promise实例
Promise.resolve()用法
Promise.resolve(“成功状态”).then(value=>{console.log(value)})
等价于
new Promise(resolve=>resolve("成功状态"))
Promise.reject()
将现有对象转换为 一个失败态的Promise实例
Promise.reject()用法
Promise.reject("失败状态").catch(error=>{console.log(error)})
等价于
new Promise((resolve,reject)=>reject("失败状态"))
Promise.all()
const p = Promise.all( [p1,p2,p3...] );
将多个Promise实例包装成一个新的Promise实例,
p1,p2,p3都是Promise实例,
如果不是,内部会先调用Promise.resolve()方法,将每个Promise实例包装成成功态的Promise实例,再进一步处理
内部实例全部为成功态时,才算做整个为成功态,当其中有任一为失败态时,则整体为失败态
Promise.race()
将多个Promise实例包装成一个新的Promise实例,
用法与Promise.all()相类似,但是执行不同,
Promise.race()是成功态还是失败态主要取决于内部p1,p2...哪一个先发生改变,
Promise.race()跟随内部状态最先变化的实例变化
延迟打印
function delayPrint(data){
return new Promise((resolve,reject)=>{
setTimeout(()=>{
resolve("data")
//reject("错误态")
},1000)
})
}
const p1 = delayPrint("I")
const p2 = delayPrint("like")
const p3 = delayPrint("money")
Promise.all([p1,p2,p3]).then(allData=>{
console.log(allData) //输出数组 ["I","like","money"]
})
Promise.all(["I","like","money"]).then(allData=>{
console.log(allData) //输出数组 ["I","like","money"]
})
function delayPrint(data){
return new Promise((resolve,reject)=>{
setTimeout(()=>{
resolve("data")
},1000)
})
}
const p1 = delayPrint("I")
const p2 = delayPrint("like")
const p3 = delayPrint("money")
Promise.race([p1,p2,p3]).then(data=>{
console.log(data) //输出 :1,因为1先变化
},error=>{
console.log(error)
})
function delayPrint(data){
return new Promise((resolve,reject)=>{
setTimeout(()=>{
reject("错误态")
},1000)
})
}
const p1 = delayPrint("I")
const p2 = delayPrint("like")
const p3 = delayPrint("money")
Promise.all([p1,p2,p3]).catch(error=>{
console.log(error) //输出 :错误态
})
Promise.race([p1,p2,p3]).then(data=>{
console.log(data)
},error=>{
console.log(error) //输出 :错误态
})
function delayPrint(data){
return new Promise((resolve,reject)=>{
setTimeout(()=>{
resolve("data")
},1000)
})
}
const p1 = delayPrint("I")
const p2 = delayPrint("like")
const p3 = delayPrint("money")
Promise.race([p1,p2,p3]).then(data=>{
console.log(data) //输出 :I,因为p1先变化
},error=>{
console.log(error)
})
五. Promise的应用(笔试/需求..)
1.异步加载图片
function loadmage(imgUrl){
return new Promise((resolve,reject)=>{
const image = new Image()
image.onload = resolve //加载成功执行resolve
image.onerror = reject //加载失败执行reject
img.src = imgUrl;
})
}
loadImage("xxx.png")
.then(()=>{})
.catch(()=>{})
2.代码输出顺序(js循环事件)
setTimeout(function(){
console.log(1)
},0) //宏任务
Promise.resolve().then(()=>{
console.log(2)
}) //微任务
new Promise((resolve)=>{
resolve(5) //微任务
console.log(3)
}) //同步任务
.then(n=>{
console.log(n)
}) //微任务
console.log(4)
输出顺序:3 4 2 5 1
同步=>微任务=>宏任务
边栏推荐
猜你喜欢
随机推荐
使用.NET简单实现一个Redis的高性能克隆版(一)
2022爱分析· 银行数字化厂商全景报告
ICDAR比赛技术分享
#yyds干货盘点# 面试必刷TOP101:两个链表的第一个公共结点
新特性解读 | MySQL 8.0 在线调整 REDO
软件盘点企业使用服装ERP的好处
酷开科技 × StarRocks:统一 OLAP 分析引擎,全面打造数字化的 OTT 模式
A complete detailed tutorial on building intranet penetration ngrok (with pictures and truth)
九种方法!教你如何读取resources目录下的文件路径
精酿啤酒品牌,过把瘾就死?
Atomic Wallet已支持TRC20-USDT
Adobe是什么?
China Hashpower Conference Ascension Kunpeng Ecological Forum was held; Kuaishou established an independent to B business department…
云GPU如何安装和启动VNC远程桌面服务?
【时间的比较】
PMP试题 | 每日一练,快速提分
一加Ace值得买吗?用实力诠释性能的强大
websocket Handshake failed due to invalid Upgrade header
Crack: WebKitX ActiveX and WebKitX VHX
JS string to GBK encoding ultra-reduced implementation