zoukankan      html  css  js  c++  java
  • 通过一道笔试题浅谈javascript中的promise对象

    因为前几天做了一个promise对象捕获错误的面试题目,所以这几天又重温了一下promise对象。现在借这道题来分享下一些很基础的知识点。

    下面是一个面试题目,三个promise对象捕获错误的例子,返回结果有什么不同。

    //使用throw添加错误事件
    var p = new Promise(function(resolve, reject) {
        resolve("ok");
        throw new Error('error0');
        //setTimeout(function() { throw new Error('error1') }, 0);
    });
    p.then(function(value){
        console.log(value) 
     })
     .catch(funcrion(err){
        console.log(err)
        });
    process.on('unhandledRejection', function (err, p) { console.error('catch exception:',err.stack) });
    //设置定时器来抛出错误事件
    var p = new Promise(function(resolve, reject) {
        resolve("ok");
        //throw new Error('error0');
        setTimeout(function() { throw new Error('error1') }, 0);
    });
    
    p.then(function(value){
        console.log(value) 
     }) .catch(funcrion(err){ console.log(err) })
    ;
    process.on('unhandledRejection', function (err, p) { console.error('catch exception:',err.stack) });
    //同时添加错误事件
    var p = new Promise(function(resolve, reject) {
        resolve("ok");
        throw new Error('error0');
        setTimeout(function() { throw new Error('error1') }, 0);
    });
    p.then(function(value){
        console.log(value) 
     })
     .catch(funcrion(err){
        console.log(err)
        });
    process.on('unhandledRejection', function (err, p) { console.error('catch exception:',err.stack) });

    先把问题放在这里,如果一眼能看出结果的大大们就不用再往下面读了。。

    大概在很早以前就有了解过javascript中实现异步编程的四种方式。分别是1.回调函数 2.事件监听 3.发布、订阅事件 4.promise对象

    前三种我们可以说是屡见不鲜了,回调函数,事件监听这算是js的“灵魂”了。。发布、订阅事件也是比较常见的。今天我们就来浅显学习下promise对象,以及它能实现异步编程的原理,最后是上面那个题目的答案以及我个人的一些理解。

    一、什么是promise对象,它能干什么?

    Promises对象是在CommonJS工作组提出的一种规范,目的是为异步编程提供统一接口。现已在ECMAScript2015(ES6)中实现。

    Promise 对象用于延迟(deferred) 计算和异步(asynchronous ) 计算.。一个Promise对象代表着一个还未完成,但预期将来会完成的操作。

    Promise 对象是一个返回值的代理,这个返回值在promise对象创建时未必已知。它允许你为异步操作的成功或失败指定处理方法。 这使得异步方法可以像同步方法那样返回值:异步方法会返回一个包含了原返回值的 promise 对象来替代原返回值。

    Promise对象有以下几种状态:

    pending: 表示一个初始状态, 非 fulfilled 或 rejected。

    fulfilled: 成功的操作。

    rejected: 失败的操作。

    每一个异步任务都会返回一个Promise对象,该对象有一个then方法,允许指定回调函数。可以根据Promise对象的状态相应的去执行对应的回调函数。我们大概了解了promise存在的意义,下面我们具体去看一下该对象常用的几个API。

    二、常用的API

    1.Promise.prototype.then()

    promise实例具有then方法,因此then方法是定义在原型对象promise.prototype上的。它的作用是为promise实例添加状态改变时的回调函数。

    then()方法的第一个参数是resolved状态的回调函数,第二个参数(可选)是rejected状态的回调函数。then方法返回的是一个新的promise实例,非原来的那个promise实例,因此可以采用链式写法,then方法之后还可调用一个then方法。

    var p=new Promise(function(resolve,eject){
        resolve("ok");    
    });
    p.then(function(value){console.log(val)},
     function(err)(console.log(err))       
    );

    then方法的第二个参数一般不推荐写。有以下两个原因:第一个原因,由于是链式操作,这个then方法之后还可能会有其他操作,如果此时把错误捕捉的函数放在后面方法前边的话,并且之后再无错误捕获方法,then之后的错误就会捕捉不到。第二个原因是在then方法里面,两个参数都是回调函数写了一大堆,这样结构看起来比较混乱。

    所以下面就有了这个方法,一般写在链式写法的最后。这样就可以捕获到前面所有的错误。

    2.Promise.prototype.catch()

    这个方法是.then(null,rejection)的别名,这也能看出这个方法是专门只能用来捕获错误信息,用于指定发生错误时的回调函数。

    但是使用这个这个方法的时候要注意一下几点:

    (1)当promise状态已经变成resolved的时候,再抛出错误时是无效的。看下面的代码。

    var promise=new promise(function(resolve,reject){
       resolve("ok");
       throw new Error("test");      
    });
    promise.then(function(value){consloe.log(val); })
                .catch(function(error){console.log(err)});

    promise状态在resolve("ok");之后就会把promise的状态变为resolved,之后抛出错误也不会把promise状态变为rejected,所以catch方法并不会捕获到错误。

    Promise 对象的状态改变,只有两种可能:从 Pending 变为 Resolved 和从 Pending 变为 Rejected。只要这两种情况有任意一种发生,状态就相当于凝固了,不会再变了,会一直保持这个结果。

     (2)尽量将catch方法写在链式操作的最后,原因上面都已经说过了,也正是捕获错误不推荐写then方法的原因之一。错误会一直冒泡到最后,catch放在最后会捕捉到所有错误。当catch设置的过早,并且之后在没有catch方法的话,那么这个catch之后发生的错误不会被捕获到。

    (3)当没有使用catch方法指定错误处理函数的回调函数时,promise对象里面抛出的错误不会传递到外层的代码。

    3.Promise.resolve()

    这个方法的作用就是将现有的对象转化为Promise对象,进而可以执行这些方法。

    Promise.resolve("foo");
    
    //这就相当于下面这种写法
    
    new Promise(function(resolve){
       resolve("foo");     
    });

    4.Promise.all()

    这个方法用于将多个promise实例,包装成一个新的promise实例。

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

    p1,p2,p3都是promise对象的实例,如果不是的话,则会调用Promise.resolve()方法,将参数转化为Promise实例,之后再继续进行进一步的处理。

    并且要注意一下两点:

    (1)只有当p1,p2,p3状态都变为fulfilled之后,p的状态才会变为fulfilled。

    (2)只要p1.p2,p3中有任意一个状态变为rejected,p的状态就会变为rejected。

    三、实现异步编程的原理

    大概原理就是正如它们所说Promise对象相当于是一个状态机,在其内部使用resolve方法,使其由初始状态变为成功时的fulfilled状态或者执行失败后的rejected状态。这时内部的工作就完成了,开始由外部监听其内部的状态的改变,调用then()方法(catch()方法相当于then内部的第二个参数方法)对应的状态调用对应的处理函数。这样就大概是Promise对象实现异步编程的原理。

    四、开头所给问题的答案

    我们可以看到三个实例外部函数的写法一模一样,不同的是Promise对象内部抛出错误所使用的方法。

    第一个使用throw抛出错误,理应被外部的catch方法所捕获到。但是但是。。之前已经说过Promise对象状态一旦被改变之后就“凝固”了,一旦执行

    resolve("ok");

    状态被设定为fulfilled之后,再进行抛出错误处理,错误也不会被后续的catch方法捕获到。所以这里只会去执行then()方法里面的内容。即只会打印出 “ok”。

    第二个通过定时器抛出一个错误。这里虽然状态已经变为fulfilled,但是定时器抛出的错误属于异步抛出的错误,无法被try catch捕获到,因此和Promise对象无关,所以错误可以正常的抛出来,所以这里的答案应该是先打印出“ok”,之后抛出process里面定义的错误。

    第三个同时使用throw和定时器抛出错误。是不是里应当和第二个执行情况一样吗?这样想就错了。当执行throw之后,虽然错误未被外部函数捕获处理,但这也是个实实在在存在的错误啊,对于javascript来说,有错就不会继续往下面执行了。所以并不会执行到定时器抛出错误就停止了。因此这个问题的答案应该是 只打印出“ok”。

    嗯,结束。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。。

  • 相关阅读:
    CodeForces 656B
    时间限制
    哈哈
    &1的用法
    codeforces 385 c
    hdu 1176 免费馅饼
    poj 1114 完全背包 dp
    poj 1115 Lifting the Stone 计算多边形的中心
    jar包解压
    重定向
  • 原文地址:https://www.cnblogs.com/huangzhilong/p/5358493.html
Copyright © 2011-2022 走看看