ESCMScript6(3)Promise对象
1. Promise的含义
Promise
是异步编程的一种解决方案,比传统的解决方案——回调函数和事件——更合理和更强大。它由社区最早提出和实现,ES6
将其写进了语言标准,统一了用法,原生提供了Promise
对象。
所谓Promise
,简单说就是一个容器,里面保存着某个未来才会结束的事件(通常是一个异步操作)的结果。从语法上说,Promise
是一个对象,从它可以获取异步操作的消息。Promise
提供统一的 API
,各种异步操作都可以用同样的方法进行处理。
Promise对象有以下两个特点
对象的状态不受外界影响。
Promise
对象代表一个异步操作,有三种状态:pending
(进行中)、fulfilled
(已成功)和rejected
(已失败)。只有异步操作的结果,可以决定当前是哪一种状态,任何其他操作都无法改变这个状态。这也是Promise
这个名字的由来,它的英语意思就是“承诺”,表示其他手段无法改变。一旦状态改变,就不会再变,任何时候都可以得到这个结果。
Promise
对象的状态改变,只有两种可能:从pending
变为fulfilled
和从pending
变为rejected
。只要这两种情况发生,状态就凝固了,不会再变了,会一直保持这个结果,这时就称为resolved
(已定型)。如果改变已经发生了,你再对Promise
对象添加回调函数,也会立即得到这个结果。这与事件(Event)
完全不同,事件的特点是,如果你错过了它,再去监听,是得不到结果的。
注意,为了行文方便,本章后面的resolved
统一只指fulfilled
状态,不包含rejected
状态。
有了Promise
对象,就可以将异步操作以同步操作的流程表达出来,避免了层层嵌套的回调函数。此外,Promise
对象提供统一的接口,使得控制异步操作更加容易。
Promise的缺点
Promise
也有一些缺点。
首先,无法取消Promise
,一旦新建它就会立即执行,无法中途取消。
其次,如果不设置回调函数,Promise
内部抛出的错误,不会反应到外部。
第三,当处于pending
状态时,无法得知目前进展到哪一个阶段(刚刚开始还是即将完成)。
如果某些事件不断地反复发生,一般来说,使用 Stream
模式是比部署Promise
更好的选择。
2.基本用法
一个promise
可以通过promise
构造函数来创建,这个构造函数只接受一个参数:包含初始化promise
代码的执行器(executor)
函数,在该函数内包含需要异步执行的代码。执行器函数接受两个参数,分别是resolve
函数和reject
函数,这两个函数由JavaScript
引擎提供,不需要我们自己编写。异步操作结束成功时调用resolve
函数,失败时调用reject
函数。
示例代码如下:
const promise = new Promise((resolve, reject) => {
// 开启异步操作
setTimeout(function() {
try {
let c = 6 / 2
// 执行成功时调用resolve函数
resolve(c)
} catch (ex) {
// 执行失败时调用reject函数
reject(ex)
}
}, 1000)
})
在执行器函数内包含了异步调用,在1s
后执行两个数的除法运算,如果成功,则用相除的结果作为参数调用resolve
函数,失败则调用reject
函数。
每个promise
都会经历一个短暂的生命周期:先是出于进行中(pending)
的状态,此时操作尚未完成,所以它是未处理的,一旦异步操作执行结束,promise
则变为已处理的状态。操作结束后,根据异步操作执行成功与否,可以进入以下两个状态之一:
(1)fulfilled
:promise
异步操作成功完成
(2)rejected
:由于程序错误或者其他的一些原因,promise
异步操作未能成功完成,即已失败。
一旦promise
状态改变,就不会再变,任何时候都可以得到这个结果。promise
对象的状态改变,只有两种可能:从pending
变为fulfilled
和从pending
变为rejected
。
在promise
状态改变后,我们怎么去根据不同的状态来做相应的处理呢?promise
对象有一个then()
方法,它接受2个参数:第一个是当promise
的状态变为fulfilled
时调用的函数,与异步操作相关的附加数据通过调用resolve
函数产地给这个完成函数;第二个是当promise
的状态变为rejected
时要调用的函数,所有与失败相关的附加数据通过调用rejected
函数传递个这个拒绝函数。添加promise
的then()
方法的调用,代码如下:
promise.then(value => {
console.log(value); // 3
}, err => {
console.error(err.message);
})
then()
方法的两个参数都是可选的。例如,只在执行失败后进行处理,可以给then()
方法的第一个参数出传递null
。代码如下:
promise.then(null, err => {
console.error(err.message);
})
promise
对象还有一个catch()
方法,用于在执行失败后进行处理,等价于上述只给then()
方法传入拒绝处理函数的代码,如下:
promise.catch(err => {
console.error(err.message)
})
但是通常我们是将then()
方法和catch()
方法一起使用来对异步操作的结果进行处理,这样能更清楚的指明操作结果是成功还是失败,代码如下:
promise.then(value => {
// 完成
console.log(value);
}).catch(err => {
// 拒绝
console.error(err.message);
})
ESCMScript6(3)Promise对象的更多相关文章
- angular学习笔记(二十八-附2)-$http,$resource中的promise对象
下面这种promise的用法,我从第一篇$http笔记到$resource笔记中,一直都有用到: HttpREST.factory('cardResource',function($resource) ...
- ES6深入学习记录(二)promise对象相关
1.Promise的含义 Promise是异步编程的一种解决方案,比传统的解决方案--回调函数和事件更合理和强大.ES6将其写进了语言标准,统一了用法,原生提供了promise对象. 所谓Promis ...
- es6中的promise对象
Promise是异步里面的一种解决方案,解决了回调嵌套的问题,es6将其进行了语言标准,同意了用法,提供了`promise`对象, promise对象有三种状态:pending(进行中) .Resol ...
- ES6的promise对象应该这样用
ES6修补了一位Js修真者诸多的遗憾. 曾几何时,我这个小白从js非阻塞特性的坑中爬出来,当我经历了一些回调丑陋的写法和优化的尝试之后,我深深觉得js对于多线程阻塞式的开发语言而言,可能有着其太明显的 ...
- Angularjs promise对象解析
1.先来看一段Demo,看完这个demo你可以思考下如果使用$.ajax如何处理同样的逻辑,使用ng的promise有何优势? var ngApp=angular.module('ngApp',[]) ...
- JavaScript异步编程(1)- ECMAScript 6的Promise对象
JavaScript的Callback机制深入人心.而ECMAScript的世界同样充斥的各种异步操作(异步IO.setTimeout等).异步和Callback的搭载很容易就衍生"回调金字 ...
- Promise对象
1.Promise思想:每一个异步任务立刻返回一个Promise对象,由于是立刻返回,所以可以采用同步操作的流程.这个Promises对象有一个then方法,允许指定回调函数,在异步任务完成后调用. ...
- angularJS中的Promise对象($q)的深入理解
原文链接:a better way to learn AngularJS - promises AngularJS通过内置的$q服务提供Promise编程模式.通过将异步函数注册到promise对象, ...
- 通过一道笔试题浅谈javascript中的promise对象
因为前几天做了一个promise对象捕获错误的面试题目,所以这几天又重温了一下promise对象.现在借这道题来分享下一些很基础的知识点. 下面是一个面试题目,三个promise对象捕获错误的例子,返 ...
随机推荐
- MindSpore Lite整体架构介绍
MindSpore Lite整体架构介绍 MindSpore Lite框架的总体架构如下所示: 前端(Frontend): 负责模型生成,用户可以通过模型构建接口构建模型,将第三方模型和MindSpo ...
- ADAS系统长篇综述(下)
ADAS系统长篇综述(下) 四.ADAS架构设计的进化阶梯 前面谈到的产品的商业化推广渗透和产品的功能演进渗透,目的是让大家去概念化.当然,最后的赢家一定是实干者,能够在具体技术实现路径上进行深度耕耘 ...
- CodeGen概述
CodeGen概述 CodeGen是在协同开发环境中工作的软件开发人员可以用来生成源代码的工具.该代码可能是Synergy DBL代码,也可能是其他语言的源代码.CodeGen并不局限于为任何特定的开 ...
- 小白自制Linux开发板 一. 瞎抄原理图与乱画PCB
因为墨云是基于高中物理水平的电路知识来学习.而且此前也就玩过树莓派.Esp8266之类的开发板,水平基础趋近于零,所以在写这个系列的时候抱着记录的心态.还望不足之处还望大佬们指正. <论语> ...
- eclipse左边的工程列表窗口不见了解决方案
解决eclipse左边的工程列表窗口看不到工程目录的方法: Window->Show View->Project Explorer(如果没有Project Explorer选项,则Wind ...
- UF_UI 界面相关
Open C uc1600uc1601uc1603 uc1605uc1607uc1608uc1609uc1613 获取用户输入的字符串uc1615uc1616uc1617uc1618uc163 ...
- 【NX二次开发】 获取产品曲面上多个点对应的面的垂直矢量!
说明:选择一个产品面,选择面上的点,生成点在此面上的法线反向,生成直线.生成矢量的起点坐标,和矢量方向信息.可用于三坐标测量,如果需要可以自己编个插件用! 效果图: 源码: //----------- ...
- 孟老板 ListAdapter封装, 告别Adapter代码 (三)
BaseAdapter系列 ListAdapter封装, 告别Adapter代码 (一) ListAdapter封装, 告别Adapter代码 (二) ListAdapter封装, 告别Adapter ...
- open数据库报错ERROR at line 1: ORA-03113: end-of-file on communication channel Process ID: 3880 Session ID: 125 Serial number: 3
1.今天打开数据时,失败,报错 ERROR at line 1:ORA-03113: end-of-file on communication channelProcess ID: 3880Sessi ...
- expdp数据泵导出数据汇总
[oracle@enmo1 ~]$ mkdir datadump[oracle@enmo1 ~]$ cd datadump/[oracle@enmo1 datadump]$ pwd/home/orac ...