Promise简介
Promise是ES6中的函数,规范了如何处理异步任务的回调函数,功能类似于jQuery的defferred。简单说就是通过promise对象的不同状态调用不同的回调函数。目前IE8及以下不支持,其他浏览器都支持。
promise对象的状态,从Pending转换为Resolved或Rejected之后,这个promise对象的状态就不能再发生任何变化。
使用步骤:
- var promise = new Promise(function(resolve, reject) {
- //定义异步执行的任务(如setTimeout、ajax请求等),该任务会立即执行。
//在任务中调用resolve(value) 或 reject(error)方法,以改变promise对象的状态。改变状态的方法只能在异步任务中调用。- //promise状态改变后,会调用对应的回调方法
- setTimeout(function(){
- //do something else...
- resolve('callback registerd in then')
- },1000)
- });
- promise.then(function(value){
- //resolve时的回调函数,参数由异步的函数传进来
- alert(value+' is invoked');
- })
- .catch(function(error){
- //发生异常时或明确reject()时的回调函数
- })
具体使用:
- function getURL(URL) { //因为promise创建时即执行,所以用工厂函数封装promise对象
- return new Promise(function (resolve, reject) {
- var req = new XMLHttpRequest();
- req.open('GET', URL, true);
- req.onload = function () {
- if (req.status === 200) {
- resolve(req.responseText);
- } else {
- reject(new Error(req.statusText));
- }
- };
- req.onerror = function () {
- reject(new Error(req.statusText));
- };
- req.send();
- });
- }
- // 运行示例
- var URL = "http://httpbin.org/get";
- getURL(URL).then(function onFulfilled(value){
- console.log(value);
- }).catch(function onRejected(error){
- console.error(error);
- });
Promise的回调只有异步方式,即使是同步任务的回调也是异步执行 。
- var promise = new Promise(function (resolve){
- console.log("inner promise"); // 执行1:Promise封装的任务先执行
- resolve(‘callBack’);
- });
- promise.then(function(value){
- console.log(value); // 执行3:虽然注册时状态为resolved,但回调仍是异步的;
- });
- console.log("outer promise"); // 执行2:同步代码先执行
promise的方法链
then方法注册的回调会依次被调用,每个then方法之间通过return 返回值传递参数。但是回调中的异常会导致跳过之间then的回调,直接调用catch的回调,之后再继续调用剩下的then的回调。在then(onFulfilled, onRejected)中,onFulfilled的异常不会被自己的onRejected捕获,所以优先使用catch,而不要使用onRejected注册回调。
promise .then(taskA) .then(taskB) .catch(onRejected) .then(finalTask);
taskA抛异常,taskB被跳过,finalTask仍会被调用,因为catch返回的promise对象的状态为resolved。
then方法内可以返回3种值
1. 返回另一个promise对象,下一个then方法根据其状态选择onFullfilled/onRejected回调函数执行,参数仍由新promise的resolv/reject方法传递;
- var promise = new Promise(function(resolve, reject) {
- setTimeout(function(){
- resolve('触发第一个then中的回调')
- },1000)
- })
- .then(function(value){
- console.log(value);
- //返回一个新Promise对象,该对象的状态决定其后的then中的回调啥时候执行
- //如果返回undefined或其他非Promise对象,则会同步执行第二个then
- return new Promise(function(resolve,reject){
- setTimeout(function(){resolve('触发第二个then中的回调')},2000)
- })
- })
- .then(function(v){
- console.log(v);
- })
2. 返回一个同步值,下一个then方法沿用当前promise对象的状态,无需等异步任务结束会立即执行;实参为上一then的返回值;如果没有return,则默认返回undefined;
3. 抛出异常(同步/异步):throw new Error(‘xxx’);
then不仅是注册一个回调函数,还会将回调函数的返回值进行变换,创建并返回一个新promise对象。实际上Promise在方法链中的操作的都不是同一个promise对象。
- var aPromise = new Promise(function (resolve) {
- resolve(100);
- });
- var thenPromise = aPromise.then(function (value) {
- console.log(value);
- });
- var catchPromise = thenPromise.catch(function (error) {
- console.error(error);
- });
- console.log(aPromise !== thenPromise); // => true
- console.log(thenPromise !== catchPromise);// => true
Promise构造方法中显式return的promis对象的状态,不会被传递给then方法。
- //少用此种方式
- new Promise(function () {
- return Promise.resolve(); //即使内部返回的promise对象的状态为resolved,依然不能传给后面的then方法。
- }).then(function (value) {
- console.log("Can't be invoked"); //不会被调用,前者的pomise一直处于pending状态
- })
//正常模式- Promise.resolve().then(function () {
- return Promise.resolve(); //then方法中可以传递promise对象的状态给后面的then使用
- }).then(function (value) {
- console.log("invoked"); //会被调用
- })
Promise.all()静态方法,同时进行多个异步任务。在接收到的所有promise对象都变为FulFilled 或者Rejected 状态之后才会继续进行后面的处理。
Promise.all([promiseA, promiseB]).then(function(results){//results是个数组,元素值和前面promises对象对应});
- // 由promise对象组成的数组会同时执行,而不是一个一个顺序执行,开始时间基本相同。
- function timerPromisefy(delay) {
- console.log('开始时间:”'+Date.now())
- return new Promise(function (resolve) {
- setTimeout(function () {
- resolve(delay);
- }, delay);
- });
- }
- var startDate = Date.now();
- Promise.all([
- timerPromisefy(100), //promise用工厂形式包装一下
- timerPromisefy(200),
- timerPromisefy(300),
- timerPromisefy(400)
- ]).then(function (values) {
- console.log(values); // [100,200,300,400]
- });
不同时执行,而是一个接着一个执行promise
- //promise factories返回promise对象,只有当前异步任务结束时才执行下一个then
- function sequentialize(promiseFactories) {
- var chain = Promise.resolve();
- promiseFactories.forEach(function (promiseFactory) {
- chain = chain.then(promiseFactory);
- });
- return chain;
- }
Promise.race()同all()类似,但是race()只要有一个promise对象进入 FulFilled 或者 Rejected 状态的话,就会执行对应的回调函数。不过在第一个promise对象变为Fulfilled之后,并不影响其他promise对象的继续执行。
- //沿用Promise.all()的例子
- Promise.race([
- timerPromisefy(1),
- timerPromisefy(32),
- timerPromisefy(64),
- timerPromisefy(128)
- ]).then(function (value) {
- console.log(values); // [1]
- });
Promise.race()作为定时器的妙用
- Promise.race([
- new Promise(function (resolve, reject) {
- setTimeout(reject, 5000); // timeout after 5 secs
- }),
- doSomethingThatMayTakeAwhile()
- ]);
在then中改变promise状态
因为then的回调中只有value参数,没有改变状态的方法(只能在构造方法的异步任务中使用),要想改变传给下一个then的promise对象的状态,只能重新new一个新的Promise对象,在异步任务中判断是否改变状态,最后return出去传给下一个then/catch。
- var promise = Promise.resolve('xxx');//创建promise对象的简介方法
- promise.then(function (value) {
- var pms=new Promise(function(resolve,reject){
- setTimeout(function () {
- // 在此可以判断是否改变状态reject/resolve
- reject(‘args’);
- }, 1000);
- })
- return pms; //该promise对象可以具有新状态,下一个then/catch需要等异步结束才会执行回调;如果返回普通值/undefined,之后的then/catch会立即执行
- }).catch(function (error) {
- // 被reject时调用
- console.log(error)
- });
获取两个promises的结果
- //方法1:通过在外层的变量传递
- var user;
- getUserByName('nolan').then(function (result) {
- user = result;
- return getUserAccountById(user.id);
- }).then(function (userAccount) {
- //可以访问user和userAccount
- });
- //方法2:后一个then方法提到前一个回调中
- getUserByName('nolan').then(function (user) {
- return getUserAccountById(user.id).then(function (userAccount) {
- //可以访问user和userAccount
- });
- });
注意使用promise时的整体结构
假定doSomething()和doSomethingElse()都返回了promise对象
常用方式:
- doSomething().then(doSomethingElse).then(finalHandler);
- doSomething
- |-----------------|
- doSomethingElse(resultOfDoSomething) //返回新promise,下一个then要收到新状态才执行
- |------------------|
- finalHandler(resultOfDoSomethingElse)
- |---------------------|
常用变通方式:
- doSomething().then(function () { return doSomethingElse();}).then(finalHandler);
- doSomething
- |-----------------|
- doSomethingElse(undefined) //then外层函数的arguments[0]== resultOfDoSomething
- |------------------|
- finalHandler(resultOfDoSomethingElse)
- |------------------|
错误方式1:
- doSomething().then(function () { doSomethingElse();}).then(finalHandler);
- doSomething
- |-----------------|
- doSomethingElse(undefined) //虽然doSomethingElse会返回promise对象,但最外层的回调函数是return undefined,所以下一个then方法无需等待新promise的状态,会马上执行回调。
- |------------------|
- finalHandler(undefined)
- |------------------|
错误方式2:
- doSomething().then(doSomethingElse()).then(finalHandler);
- doSomething
- |-----------------|
- doSomethingElse(undefined) //回调函数在注册时就直接被调用
- |----------|
- finalHandler(resultOfDoSomething)
- |------------------|
参考文章:https://pouchdb.com/2015/05/18/we-have-a-problem-with-promises.html
jQuery中的deferred对象同Promise对象实现相同的功能:
- // 创建deferred对象
- var dtd = $.Deferred();
- //异步任务中改变deferred对象的状态
- dtd.resolve('resolved');
- // 注册回调函数,deferred状态改变后执行回调。
- dtd.done(doneCallback [, doneCallback2])
Promise简介的更多相关文章
- 原来你是这样的Promise
1. Promise简介 promise是异步编程的一种解决方案,它出现的初衷是为了解决回调地狱的问题. 打个比方,我需要: --(延迟1s)--> 输出1 --(延迟2s)--> 输出2 ...
- Promise及Async/Await
一.为什么有Async/Await? 我们都知道已经有了Promise的解决方案了,为什么还要ES7提出新的Async/Await标准呢? 答案其实也显而易见:Promise虽然跳出了异步嵌套的怪 ...
- promise请求数据用法
Promise简介 Promise 是异步编程的一种解决方案,比传统的解决方案–回调函数和事件--更合理和更强大.ES6将其写进了语言标准,统一了语法,里面保存着某个未来才回结束的事件(通常是一个异步 ...
- 透过面试题来说说Promise
前言 我们先看看这几个来自大厂的面试题 面试题1: const promise = new Promise(function(resolve,reject){ console.log(1) resol ...
- promise介绍
promise简介 Promise的出现,原本是为了解决回调地狱的问题.所有人在讲解Promise时,都会以一个ajax请求为例,此处我们也用一个简单的ajax的例子来带大家看一下Promise是如何 ...
- 异步Promise及Async/Await最完整入门攻略
一.为什么有Async/Await? 我们都知道已经有了Promise的解决方案了,为什么还要ES7提出新的Async/Await标准呢? 答案其实也显而易见:Promise虽然跳出了异步嵌套的怪圈, ...
- 【一起来烧脑】读懂Promise知识体系
知识体系 Promise基础语法,如何处理错误,简单介绍异步函数 内容 错误处理的两种方式: reject('错误信息').then(null, message => {}) throw new ...
- 异步Promise及Async/Await可能最完整入门攻略
此文只介绍Async/Await与Promise基础知识与实际用到注意的问题,将通过很多代码实例进行说明,两个实例代码是setDelay和setDelaySecond. tips:本文系原创转自我的博 ...
- 从源码上理解Netty并发工具-Promise
前提 最近一直在看Netty相关的内容,也在编写一个轻量级的RPC框架来练手,途中发现了Netty的源码有很多亮点,某些实现甚至可以用苛刻来形容.另外,Netty提供的工具类也是相当优秀,可以开箱即用 ...
随机推荐
- PHPCMSV9上线方法及文件权限设置
上线步骤: a.替换代码和数据库文件内的域名b.修改cache/configs/database.php中的数据库密码c.修改cache/configs/system.php文件中的网站路径变量 'w ...
- 老李案例分享:Weblogic性能优化案例
老李案例分享:Weblogic性能优化案例 POPTEST的测试技术交流qq群:450192312 网站应用首页大小在130K左右,在之前的测试过程中,其百用户并发的平均响应能力在6.5秒,性能优化后 ...
- 测试开发Python培训:抓取新浪微博抓取数据-技术篇
测试开发Python培训:抓取新浪微博抓取数据-技术篇 poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.在poptest的se ...
- sleep()和wait()的区别 --- 快入睡了,突然想起一个知识点,搞完就睡
自从开了博客之后,大部分是转发的,不断的ctrl+c和ctrl+v,知识是越屯越多,吸收的却很少,后来越来越懒,直接保存链接了. 我已经认识到了自己的错误,自己查询到的这些知识,以后还是会定期保存的, ...
- quartus ii中仿真rom时遇到的问题(待完善)
1.modelsim仿真只支持.hex,并不支持.mif(Memory Initialzation File). 2.在Matlab中生成.mif文件,然后再quartus中打开,转换为hex格式后另 ...
- Android 布局(线性布局、相对布局)
一.线性布局(LinearLayout) <LinearLayout****</LinearLayout>1. orientation(布局方向)value=0 horizontal ...
- druid 连接kafuk
java -Xmx256m -Duser.timezone=UTC -Dfile.encoding=UTF-8 -Ddruid.realtime.specFile=examples/indexing/ ...
- NSUserDefaults的使用,保存登录状态和设置的轻量本地化存储
NSDictionary* defaults = [[NSUserDefaults standardUserDefaults] dictionaryRepresentation]; if([[NSUs ...
- oracle 归档日志满 报错ORA-00257: archiver error. Connect internal only, until freed
归档日志满导致无法用户无法登陆 具体处理办法 --用户登陆 Microsoft Windows [Version 6.1.7601] Copyright (c) Microsoft Corporati ...
- SQL Server 数据库连接方法
我们用c#写ado或者是asp,都需要连接数据库来读写数据,今天我们就来总结一下数据库连接都有哪些方法. 首先我们就写最直接的方法,在事件中直接连接.(在这里就用WEB页面来展示) 首先我们建立web ...