JS - Promise使用详解 侵立删

转自:http://www.hangge.com/blog/cache/detail_1638.html

 

2015年6月, ES2015(即 ECMAScript 6、ES6) 正式发布。其中 Promise 被列为正式规范,成为 ES6 中最重要的特性之一。

 

1,then()方法

简单来讲,then 方法就是把原来的回调写法分离出来,在异步操作执行完后,用链式调用的方式执行回调函数。

而 Promise 的优势就在于这个链式调用。我们可以在 then 方法中继续写 Promise 对象并返回,然后继续调用 then 来进行回调操作。

 

(1)下面通过样例作为演示,我们定义做饭、吃饭、洗碗(cook、eat、wash)这三个方法,它们是层层依赖的关系,下一步的的操作需要使用上一部操作的结果。(这里使用 setTimeout 模拟异步操作)

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

//做饭

function cook(){

    console.log('开始做饭。');

    var p = new Promise(function(resolve, reject){        //做一些异步操作

        setTimeout(function(){

            console.log('做饭完毕!');

            resolve('鸡蛋炒饭');

        }, 1000);

    });

    return p;

}

 

//吃饭

function eat(data){

    console.log('开始吃饭:' + data);

    var p = new Promise(function(resolve, reject){        //做一些异步操作

        setTimeout(function(){

            console.log('吃饭完毕!');

            resolve('一块碗和一双筷子');

        }, 2000);

    });

    return p;

}

 

function wash(data){

    console.log('开始洗碗:' + data);

    var p = new Promise(function(resolve, reject){        //做一些异步操作

        setTimeout(function(){

            console.log('洗碗完毕!');

            resolve('干净的碗筷');

        }, 2000);

    });

    return p;

}


(2)使用 then 链式调用这三个方法:

1

2

3

4

5

6

7

8

9

10

cook()

.then(function(data){

    return eat(data);

})

.then(function(data){

    return wash(data);

})

.then(function(data){

    console.log(data);

});


当然上面代码还可以简化成如下:

1

2

3

4

5

6

cook()

.then(eat)

.then(wash)

.then(function(data){

    console.log(data);

});


(3)运行结果如下:

原文:JS - Promise使用详解2(ES6中的Promise)

 

2,reject()方法

上面样例我们通过 resolve 方法把 Promise 的状态置为完成态(Resolved),这时 then 方法就能捕捉到变化,并执行“成功”情况的回调。

而 reject 方法就是把 Promise 的状态置为已失败(Rejected),这时 then 方法执行“失败”情况的回调(then 方法的第二参数)。

 

(1)下面同样使用一个样例做演示

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

//做饭

function cook(){

    console.log('开始做饭。');

    var p = new Promise(function(resolve, reject){        //做一些异步操作

        setTimeout(function(){

            console.log('做饭失败!');

            reject('烧焦的米饭');

        }, 1000);

    });

    return p;

}

 

//吃饭

function eat(data){

    console.log('开始吃饭:' + data);

    var p = new Promise(function(resolve, reject){        //做一些异步操作

        setTimeout(function(){

            console.log('吃饭完毕!');

            resolve('一块碗和一双筷子');

        }, 2000);

    });

    return p;

}

 

cook()

.then(eat, function(data){

  console.log(data + '没法吃!');

})

运行结果如下:

原文:JS - Promise使用详解2(ES6中的Promise)

 

(2)如果我们只要处理失败的情况可以使用 then(null, ...),或是使用接下来要讲的 catch 方法。

1

2

3

4

cook()

.then(nullfunction(data){

  console.log(data + '没法吃!');

})

 

3,catch()方法

(1)它可以和 then 的第二个参数一样,用来指定 reject 的回调

1

2

3

4

5

cook()

.then(eat)

.catch(function(data){

    console.log(data + '没法吃!');

});


(2)它的另一个作用是,当执行 resolve 的回调(也就是上面 then 中的第一个参数)时,如果抛出异常了(代码出错了),那么也不会报错卡死 js,而是会进到这个 catch 方法中。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

//做饭

function cook(){

    console.log('开始做饭。');

    var p = new Promise(function(resolve, reject){        //做一些异步操作

        setTimeout(function(){

            console.log('做饭完毕!');

            resolve('鸡蛋炒饭');

        }, 1000);

    });

    return p;

}

 

//吃饭

function eat(data){

    console.log('开始吃饭:' + data);

    var p = new Promise(function(resolve, reject){        //做一些异步操作

        setTimeout(function(){

            console.log('吃饭完毕!');

            resolve('一块碗和一双筷子');

        }, 2000);

    });

    return p;

}

 

cook()

.then(function(data){

    throw new Error('米饭被打翻了!');

    eat(data);

})

.catch(function(data){

    console.log(data);

});

运行结果如下:

原文:JS - Promise使用详解2(ES6中的Promise)

 

这种错误的捕获是非常有用的,因为它能够帮助我们在开发中识别代码错误。比如,在一个 then() 方法内部的任意地方,我们做了一个 JSON.parse() 操作,如果 JSON 参数不合法那么它就会抛出一个同步错误。用回调的话该错误就会被吞噬掉,但是用 promises 我们可以轻松的在 catch() 方法里处理掉该错误。

 

(3)还可以添加多个 catch,实现更加精准的异常捕获。

1

2

3

4

5

6

7

8

9

10

11

somePromise.then(function() {

 return a();

}).catch(TypeError, function(e) {

 //If a is defined, will end up here because

 //it is a type error to reference property of undefined

}).catch(ReferenceError, function(e) {

 //Will end up here if a wasn't defined at all

}).catch(function(e) {

 //Generic catch-the rest, error wasn't TypeError nor

 //ReferenceError

});

 

4,all()方法

Promise 的 all 方法提供了并行执行异步操作的能力,并且在所有异步操作执行完后才执行回调。

 

(1)比如下面代码,两个个异步操作是并行执行的,等到它们都执行完后才会进到 then 里面。同时 all 会把所有异步操作的结果放进一个数组中传给 then。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

//切菜

function cutUp(){

    console.log('开始切菜。');

    var p = new Promise(function(resolve, reject){        //做一些异步操作

        setTimeout(function(){

            console.log('切菜完毕!');

            resolve('切好的菜');

        }, 1000);

    });

    return p;

}

 

//烧水

function boil(){

    console.log('开始烧水。');

    var p = new Promise(function(resolve, reject){        //做一些异步操作

        setTimeout(function(){

            console.log('烧水完毕!');

            resolve('烧好的水');

        }, 1000);

    });

    return p;

}

 

Promise

.all([cutUp(), boil()])

.then(function(results){

    console.log("准备工作完毕:");

    console.log(results);

});


(2)运行结果如下:

原文:JS - Promise使用详解2(ES6中的Promise)

 

5,race()方法

race 按字面解释,就是赛跑的意思。race 的用法与 all 一样,只不过 all 是等所有异步操作都执行完毕后才执行 then 回调。而 race 的话只要有一个异步操作执行完毕,就立刻执行 then 回调。

注意:其它没有执行完毕的异步操作仍然会继续执行,而不是停止。

 

(1)这里我们将上面样例的 all 改成 race

1

2

3

4

5

6

Promise

.race([cutUp(), boil()])

.then(function(results){

    console.log("准备工作完毕:");

    console.log(results);

});

原文:JS - Promise使用详解2(ES6中的Promise)

 

(2)race 使用场景很多。比如我们可以用 race 给某个异步请求设置超时时间,并且在超时后执行相应的操作。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

//请求某个图片资源

function requestImg(){

    var p = new Promise(function(resolve, reject){

    var img = new Image();

    img.onload = function(){

       resolve(img);

    }

    img.src = 'xxxxxx';

    });

    return p;

}

 

//延时函数,用于给请求计时

function timeout(){

    var p = new Promise(function(resolve, reject){

        setTimeout(function(){

            reject('图片请求超时');

        }, 5000);

    });

    return p;

}

 

Promise

.race([requestImg(), timeout()])

.then(function(results){

    console.log(results);

})

.catch(function(reason){

    console.log(reason);

});

上面代码 requestImg 函数异步请求一张图片,timeout 函数是一个延时 5 秒的异步操作。我们将它们一起放在 race 中赛跑。

  • 如果 5 秒内图片请求成功那么便进入 then 方法,执行正常的流程。
  • 如果 5 秒钟图片还未成功返回,那么则进入 catch,报“图片请求超时”的信息。

原文:JS - Promise使用详解2(ES6中的Promise)


原文出自:www.hangge.com  转载请保留原文链接:http://www.hangge.com/blog/cache/detail_1638.html#

 

对js Promise理解

1.优点和缺点
可以将异步操作以同步操作的流程表达出来,避免了层层嵌套的回调函数。此外,Promise对象提供统一的接口,使得控制异步操作更加容易。

Promise也有一些缺点。首先,无法取消Promise,一旦新建它就会立即执行,无法中途取消。其次,如果不设置回调函数,Promise内部抛出的错误,不会反应到外部。当处于Pending状态时,无法得知目前进展到哪一个阶段(刚刚开始还是即将完成)。

2.规范
Promise 对象有三种状态: Pending – Promise对象的初始状态,等到任务的完成或者被拒绝;Resolved – 任务执行完成并且成功的状态;Rejected – 任务执行完成并且失败的状态;
Promise的状态只可能从Pending状态转到Resolved状态或者Rejected状态,而且不能逆向转换,同时Resolved状态和Rejected状态也不能相互转换;
Promise对象必须实现then方法,then是promise规范的核心,而且then方法也必须返回一个Promise对象,同一个Promise对象可以注册多个then方法,并且回调的执行顺序跟它们的注册顺序一致;
then方法接受两个回调函数,它们分别为:成功时的回调和失败时的回调;并且它们分别在:Promise由Pending状态转换到Resolved状态时被调用和在Promise由Pending状态转换到Rejected状态时被调用。
 

3.特性

3-1.立即执行性

 

 
  1. var p=new Promise(function(resolve,reject)(){

  2. console.log("create new promise");

  3. resolve("success");

  4. });

  5.  
  6. console.log("after new promise");

  7.  
  8. p.then(function(value){

  9. console.log(value);

  10. });

  11. //create new promise

  12. //after new promise

  13. //success

 

3-2.状态不可逆性、链式调用、

 
  1. var p = new Promise(function(resolve, reject){

  2. resolve(1);

  3. });

  4. p.then(function(value){ //第一个then

  5. console.log(value);

  6. return value*2;

  7. }).then(function(value){ //第二个then

  8. console.log(value);

  9. }).then(function(value){ //第三个then

  10. console.log(value);

  11. return Promise.resolve('resolve');

  12. }).then(function(value){ //第四个then

  13. console.log(value);

  14. return Promise.reject('reject');

  15. }).then(function(value){ //第五个then

  16. console.log('resolve: '+ value);

  17. }, function(err){

  18. console.log('reject: ' + err);

  19. })

  20. //1

  21. //2

  22. //undefined

  23. //resolve

  24. //reject: reject

 

3-3.then() 回调异步性

 

 
  1. var p = new Promise(function(resolve, reject){

  2. resolve("success");

  3. });

  4.  
  5. p.then(function(value){

  6. console.log(value);

  7. });

  8.  
  9. console.log("first");

  10. //"first"

  11. //"success"

 

4.用法

 

 
  1. function getURL(URL) {

  2. return new Promise(function (resolve, reject) {

  3. var req = new XMLHttpRequest();

  4. req.open('GET', URL, true);

  5. req.onload = function () {

  6. if (req.status === 200) {

  7. resolve(req.responseText);

  8. } else {

  9. reject(new Error(req.statusText));

  10. }

  11. };

  12. req.onerror = function () {

  13. reject(new Error(req.statusText));

  14. };

  15. req.send();

  16. });

  17. }

  18. // 运行示例

  19. var URL = "http://httpbin.org/get";

  20. getURL(URL).then(function onFulfilled(value){

  21. console.log(value);

  22. }).catch(function onRejected(error){

  23. console.error(error);

  24. });

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值