[转]JS - Promise使用详解1(基本概念、使用优势)

1、promises相关概念

promises 的概念是由 CommonJS 小组的成员在 Promises/A 规范中提出来的。
 

1,then()方法介绍

根据 Promise/A 规范,promise 是一个对象,只须要 then 这一个方法。then 方法带有以下三个参数:
  • 成功回调
  • 失败回调
  • 前进回调(规范没有要求包括前进回调的实现,可是不少都实现了)。
一个全新的 promise 对象从每一个 then 的调用中返回。
 

2,Promise对象状态

Promise 对象表明一个异步操做,其不受外界影响,有三种状态:
  • Pending(进行中、未完成的)
  • Resolved(已完成,又称 Fulfilled)
  • Rejected(已失败)。
(1)promise 从未完成的状态开始,若是成功它将会是完成态,若是失败将会是失败态。
(2)当一个 promise 移动到完成态,全部注册到它的成功回调将被调用,并且会将成功的结果值传给它。另外,任何注册到 promise 的成功回调,将会在它已经完成之后当即被调用。
(3)一样的,当一个 promise 移动到失败态的时候,它调用的是失败回调而不是成功回调。
(4)对包含前进特性的实现来讲,promise 在它离开未完成状态之前的任什么时候刻,均可以更新它的 progress。当 progress 被更新,全部的前进回调(progress callbacks)会被传递以 progress 的值,并被当即调用。前进回调被以不一样于成功和失败回调的方式处理;若是你在一个 progress 更新已经发生之后注册了一个前进回调,新的前进回调只会在它被注册之后被已更新的 progress 调用。
(5)注意:只有异步操做的结果,能够决定当前是哪种状态,任何其余操做都没法改变这个状态。
 

3,Promise/A规范图解

原文:JS - Promise使用详解1(基本概念、使用优势)

4,目前支持Promises/A规范的库

  • Q:能够在NodeJS 以及浏览器上工做,与jQuery兼容,能够经过消息传递远程对象。
  • RSVP.js:一个轻量级的库,它提供了组织异步代码的工具。
  • when.js:体积小巧,使用方便。
  • NodeJS的Promise
  • jQuery 1.5:听说是基于“CommonJS Promises/A”规范
  • WinJS / Windows 8 / Metro
 

2、使用promises的优点

1,解决回调地狱(Callback Hell)问题

(1)有时咱们要进行一些相互间有依赖关系的异步操做,好比有多个请求,后一个的请求须要上一次请求的返回结果。过去常规作法只能 callback 层层嵌套,但嵌套层数过多的话就会有 callback hell 问题。好比下面代码,可读性和维护性都不好的。
1
2
3
4
5
6
7
8
9
10
11
12
firstAsync( function (data){
     //处理获得的 data 数据
     //....
     secondAsync( function (data2){
         //处理获得的 data2 数据
         //....
         thirdAsync( function (data3){
               //处理获得的 data3 数据
               //....
         });
     });
});
 
(2)若是使用 promises 的话,代码就会变得扁平且更可读了。前面提到 then 返回了一个 promise,所以咱们能够将 then 的调用不停地串连起来。其中 then 返回的 promise 装载了由调用返回的值。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
firstAsync()
.then( function (data){
     //处理获得的 data 数据
     //....
     return  secondAsync();
})
.then( function (data2){
     //处理获得的 data2 数据
     //....
     return  thirdAsync();
})
.then( function (data3){
     //处理获得的 data3 数据
     //....
});

2,更好地进行错误捕获 

多重嵌套 callback 除了会形成上面讲的代码缩进问题,更可怕的是可能会形成没法捕获异常或异常捕获不可控。
(1)好比下面代码咱们使用 setTimeout 模拟异步操做,在其中抛出了个异常。但因为异步回调中,回调函数的执行栈与原函数分离开,致使外部没法抓住异常。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
function  fetch(callback) {
     setTimeout(() => {
         throw  Error( '请求失败' )
     }, 2000)
}
 
try  {
     fetch(() => {
         console.log( '请求处理' // 永远不会执行
     })
catch  (error) {
     console.log( '触发异常' , error)  // 永远不会执行
}
 
// 程序崩溃
// Uncaught Error: 请求失败
 
(2)若是使用 promises 的话,经过 reject 方法把 Promise 的状态置为 rejected,这样咱们在 then 中就能捕捉到,而后执行“失败”状况的回调。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
function  fetch(callback) {
     return  new  Promise((resolve, reject) => {
         setTimeout(() => {
              reject( '请求失败' );
         }, 2000)
     })
}
 
 
fetch()
.then(
     function (data){
         console.log( '请求处理' );
         console.log(data);
     },
     function (reason, data){
         console.log( '触发异常' );
         console.log(reason);
     }
);

固然咱们在 catch 方法中处理 reject 回调也是能够的。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
function  fetch(callback) {
     return  new  Promise((resolve, reject) => {
         setTimeout(() => {
              reject( '请求失败' );
         }, 2000)
     })
}
 
 
fetch()
.then(
     function (data){
         console.log( '请求处理' );
         console.log(data);
     }
)
. catch ( function (reason){
     console.log( '触发异常' );
     console.log(reason);
});


原文出自:www.hangge.com  转载请保留原文连接:https://www.hangge.com/blog/cache/detail_1635.htmlhtml

相关文章
相关标签/搜索