ES6学习笔记(9)——Promise对象

落日映苍穹つ 2022-11-27 05:26 237阅读 0赞

Promise对象

在这里插入图片描述

Promise 是异步编程的一种解决方案。所谓Promise,简单说就是一个容器,里面保存着某个未来才会结束的事件的结果。Promise 是一个对象,从它可以获取异步操作的消息。Promise 提供统一的 API,各种异步操作都可以用同样的方法进行处理。可以将异步操作以同步操作的流程表达出来,避免了层层嵌套的回调函数。

特点

  1. 对象的状态不受外界影响。Promise有三个状态:Pending(进行中)、Resolved(已完成)和Rejected(已失败),只有异步操作的结果,可以决定当前是哪一种状态。
  2. 一旦状态改变,就不会再变,任何时候都可以得到这个结果。Promise对象的状态只有两种改变:从Pending变为Resolved和从Pending变为Rejected。只要其中一个情况发生,状态就会一直保持这个结果。如果改变已经发生了,你再对Promise对象添加回调函数,也会立即得到这个结果
  3. 无法取消Promise,一旦新建它就会立即执行;如果不设置回调函数,Promise内部抛出的错误,不会反应到外部;当处于Pending状态时,无法得知目前进展到哪一个阶段

基本用法

ES6规定,Promise对象是一个构造函数,用来生成Promise实例

  1. //promise构建函数接受一个函数作为参数,
  2. //函数内有两个参数:resolve和reject,它们是两个函数,由js引擎提供
  3. var promise = new Promise(function(resolve, reject) {
  4. // ... some code
  5. if (/* 异步操作成功 */){
  6. resolve(value); //状态从“未完成”到“成功”、异步操作成功时调用,并将结果作为参数抛出
  7. } else {
  8. reject(error); //状态从“未完成”变为“失败”、异步操作失败时调用,并将错误作为参数抛出
  9. }
  10. });

Promise实例生成以后,可以用then方法分别指定Resolved状态和Reject状态的回调函数

  1. //接受两个回调函数作为参数:一个状态变为Resolved时调用,一个是状态变为Reject时调用。其中,第二个函数是可选的。这两个函数都接受Promise对象传出的值作为参数
  2. promise.then(function(value) {
  3. // success
  4. }, function(error) {
  5. // failure
  6. });

一个例子

  1. //一个返回Promise实例的方法
  2. function timeout(ms) {
  3. return new Promise((resolve, reject) => {
  4. setTimeout(resolve, ms, 'done');
  5. });
  6. }
  7. //调用方法,当promise的状态变成Resolved时,就会触发then绑定的回调函数
  8. timeout(100).then((value) => { //then后面状态变为Resolved时的回调,
  9. console.log(value);
  10. });

异步加载图片的例子

  1. function loadImageAsync(url) {
  2. return new Promise(function(resolve, reject) {
  3. var image = new Image();
  4. image.onload = function() {
  5. resolve(image);
  6. };
  7. image.onerror = function() {
  8. reject(new Error('Could not load image at ' + url));
  9. };
  10. image.src = url;
  11. });
  12. }

用Promise对象实现的Ajax操作的例子

  1. //getJSON是对XMLHttpRequest对象的封装,用于发出一个针对JSON数据的HTTP请求
  2. var getJSON = function(url) {
  3. var promise = new Promise(function(resolve, reject){
  4. var client = new XMLHttpRequest();
  5. client.open("GET", url);
  6. client.onreadystatechange = handler;
  7. client.responseType = "json";
  8. client.setRequestHeader("Accept", "application/json");
  9. client.send();
  10. function handler() {
  11. if (this.readyState !== 4) {
  12. return;
  13. }
  14. if (this.status === 200) {
  15. resolve(this.response);
  16. } else {
  17. reject(new Error(this.statusText));
  18. }
  19. };
  20. });
  21. return promise;
  22. };
  23. getJSON("/posts.json").then(function(json) {
  24. console.log('Contents: ' + json);
  25. }, function(error) {
  26. console.error('出错了', error);
  27. });

状态传递:p1的状态就会传递给p2,也就是说,p1的状态决定了p2的状态

  1. var p1 = new Promise(function (resolve, reject) {
  2. // ...
  3. });
  4. var p2 = new Promise(function (resolve, reject) {
  5. // ...
  6. resolve(p1);
  7. })
  8. //一个例子
  9. var p1 = new Promise(function (resolve, reject) {
  10. setTimeout(() => reject(new Error('fail')), 3000)
  11. //2. 再2秒后,状态变成reject,传递给了p2,进而触发catch
  12. })
  13. var p2 = new Promise(function (resolve, reject) {
  14. setTimeout(() => resolve(p1), 1000) //1. 由于这里返回的是另一个Promise,所以自身的状态无效
  15. })
  16. p2
  17. .then(result => console.log(result))
  18. .catch(error => console.log(error))
  19. // 3. Error: fail

方法

  • Promise.prototype.then()

    (1) then的第一个参数是Resolved状态的回调函数,第二个参数(可选)是Rejected状态的回调函数

    (2) then方法返回的是一个新的Promise实例,因此可以采用链式写法。后面的回调会等待前一个的Promise状态发生变化后才调用

    1. //这里用then依次指定了两个回调函数。第一个回调函数完成后,会将返回结果作为参数,传入第二个回调函数
    2. getJSON("/posts.json").then(function(json) {
    3. return json.post;
    4. }).then(function(post) {
    5. // ...
    6. });
  • Promise.prototype.catch()

    (1) 用于指定发生错误时的回调函数。

    1. getJSON('/posts.json').then(function(posts) {
    2. // ...
    3. }).catch(function(error) {
    4. //(1)如果异步操作抛出错误,状态就会变为Rejected,就会调用catch方法指定的回调函数,处理这个错误
    5. //(2)then方法指定的回调函数,如果运行中抛出错误,也会被catch方法捕获。
    6. console.log('发生错误!', error);
    7. });

    (2) 相当于.then(null, rejection)

    1. p.then((val) => console.log('fulfilled:', val))
    2. .catch((err) => console.log('rejected', err));
    3. // 等同于
    4. p.then((val) => console.log('fulfilled:', val))
    5. .then(null, (err) => console.log("rejected:", err));

    (3) Promise 对象的错误会一直向后传递,直到被捕获为止。错误总是会被下一个catch语句捕获,如果catch内发生错误而后面没有catch来捕获,那么这个错误就无法被捕获到

    1. var promise = new Promise(function(resolve, reject) {
    2. reject(new Error('test'));
    3. });
    4. promise.catch(function(error) {
    5. console.log(error);
    6. });
    7. // Error: test

    (4) 两种写法对比

    1. //写法1:不能捕获前面then方法执行中的错误
    2. promise
    3. .then(function(data) {
    4. // success
    5. }, function(err) {
    6. // error
    7. });
    8. //写法2:可以捕获前面then方法执行中的错误(推荐)
    9. promise
    10. .then(function(data) { //cb
    11. // success
    12. })
    13. .catch(function(err) {
    14. // error
    15. });

    (5) catch方法返回的还是一个 Promise 对象,因此后面还可以接着调用then方法。但当没有报错时,会跳过catch

  • Promise.all()

    (1) 用于将多个Promise实例,包装成一个新的Promise实例

    1. var p = Promise.all([p1, p2, p3]);

    (2) 最终p的状态由p1、p2、p3决定;只有p1、p2、p3状态都为Resolved,p的状态才会变成Resolved;如果p1、p2、p3有一个状态为rejected,p的状态就会变成rejected

  • Promise.race()

    (1) 用于将多个Promise实例,包装成一个新的Promise实例

    1. var p = Promise.race([p1, p2, p3]);

    (2) 只要p1p2p3之中有一个实例率先改变状态,p的状态就跟着改变

  • Promise.resolve()

    (1) 将现有对象转化为Promise对象

    1. Promise.resolve('foo')
    2. // 等价于
    3. new Promise(resolve => resolve('foo'))
  • Promise.reject()

    返回一个新的 Promise 实例,该实例的状态为rejected

    1. var p = Promise.reject('出错了');
    2. // 等同于
    3. var p = new Promise((resolve, reject) => reject('出错了'))
    4. p.then(null, function (s) {
    5. console.log(s)
    6. });
    7. // 出错了
  • done():总是处于回调链的尾端,保证抛出任何可能出现的错误

    1. asyncFunc()
    2. .then(f1)
    3. .catch(r1)
    4. .then(f2)
    5. .done();
  • finally():指定不管Promise对象最后状态如何,都会执行的操作,它接受一个普通的回调函数作为参数

    1. Promise.prototype.finally = function (callback) {
    2. let P = this.constructor;
    3. return this.then(
    4. value => P.resolve(callback()).then(() => value),
    5. reason => P.resolve(callback()).then(() => { throw reason })
    6. );
    7. };

发表评论

表情:
评论列表 (有 0 条评论,237人围观)

还没有评论,来说两句吧...

相关阅读

    相关 ES6--Promise对象

    Promise对象:代表了未来某个将要发生的操作,通常是异步操作 promise对象可以将异步操作以同步的流程表达出来,避免了层层嵌套的 回调函数(回调地狱),ES6的

    相关 ES6学习笔记12 Promise

    Promise 是异步编程的一种解决方案,比传统的解决方案——回调函数和事件——更合理和更强大。 所谓Promise,简单说就是一个容器,里面保存着某个未来才会结束的事件(通

    相关 ES6学习笔记Promise

    Promise对象用于表示一个异步操作的最终状态(完成或失败)以及其返回的值。 1.回调 如果不使用Promise,在函数的一层层调用中,需要多层嵌套,这样在需求变动时

    相关 ES6--Promise对象

    概述 是异步编程的一种解决方案。从语法上说,Promise 是一个对象,从它可以获取异步操作的消息。 Promise 状态 状态的特点 Promise 异步操