Promise

1、案例

Promise用于对异步操作的进行数据处理,为异步操作的回调函数提供接口,让其在可同步执行,而不用再将回调函数进行层层嵌套。
例如,我们以下面的例子为例:

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>Index</title>
</head>
<body>
    <div id="app"></div>
    <template id="template">
        <p></p>
    </template>
</body>
<script type="text/javascript">
    var app = document.querySelector('#app');
    var p = document.getElementById('template').content.querySelector('p');
    var p1 = p.cloneNode(true);
    var p2 = p.cloneNode(true);
    var p3 = p.cloneNode(true);

    var i = 0, html;
    window.setTimeout(() => {
        p1.innerHTML = '<p>aaaaaaaa</p>';
        app.appendChild(p1);
    }, 100);
    window.setTimeout(() => {
        p2.innerHTML = '<p>bbbbbbbb</p>';
        app.appendChild(p2);
    }, 200);
    window.setTimeout(() => {
        p3.innerHTML = '<p>cccccccc</p>';
        app.appendChild(p3);
    }, 50);
</script>
</html>

对于异步函数,如果不进行回调函数嵌套,我们就无法保证其执行顺序。但是有了Promise对象之后,我们可以进行链式书写,从而保证回调执行的顺序。

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>Index</title>
</head>
<body>
    <div id="app"></div>
    <template id="template">
        <p></p>
    </template>
</body>
<script type="text/javascript">
    var app = document.querySelector('#app');
    var p = document.getElementById('template').content.querySelector('p');
    var p1 = p.cloneNode(true);
    var p2 = p.cloneNode(true);
    var p3 = p.cloneNode(true);

    var i = 0, html;
    var promise1 = new Promise((resolve, reject) => {
        window.setTimeout(() => {
            p1.innerHTML = '<p>aaaaaaaa</p>';
            resolve(p1, promise2);
        }, 100);
    });
    var promise2 = new Promise((resolve, reject) => {
        window.setTimeout(() => {
            p2.innerHTML = '<p>bbbbbbbb</p>';
            resolve(p2, promise3);
        }, 200);
    });
    var promise3 = new Promise((resolve, reject) => {
        window.setTimeout(() => {
            p3.innerHTML = '<p>cccccccc</p>';
            resolve(p3, null);
        }, 50);
    });
    promise1.then((node) => {
        app.appendChild(node);
        return promise2;
    }).then((node) => {
        app.appendChild(node);
        return promise3;
    }).then((node) => {
        app.appendChild(node);
    });
</script>
</html>

2、Promise的基本语法

Promise对象是一个代理对象(代理一个值),被代理的值在Promise对象创建时可能是未知的。它允许你为异步操作的成功和失败分别绑定相应的处理方法(handlers)。 这让异步方法可以像同步方法那样返回值,但并不是立即返回最终执行结果,而是一个能代表未来出现的结果的promise对象。

根据以上描述,我们可以知道,Promise对象可以代理一个异步处理的结果。所以Promise有fulfilled(已成功)状态和rejected(已失败)两个状态,对应Promise构造函数的两个回调函数:resolvereject

new Promise( function(resolve, reject) {...} /* executor */  );

在执行回调函数之前,Promise的状态为pedding,当执行resolve后状态变为fulfilled,执行reject之后装态变为rejected

原型方法:

//添加一个拒绝(rejection) 回调到当前 promise, 返回一个新的promise。
Promise.prototype.catch(onRejected)

//添加一个拒绝(rejection) 回调到当前 promise, 返回一个新的promise。当这个回调函数被调用,新 promise 将以它的返回值来resolve,否则如果当前promise 进入fulfilled状态,则以当前promise的完成结果作为新promise的完成结果.
Promise.prototype.then(onFulfilled, onRejected)

//无论当前promise的状态是完成(fulfilled)还是失败(rejected),都会执行
Promise.prototype.finally(onFinally)

3、Jquery中的deferred对象

Jquery中的deferred对象是原生Promise接口的实现。用法大同小异。

ajax中我们可以直接使用,因为1.5版本之后,ajax返回的值就是一个defeered对象。

旧的写法:

  $.ajax({
    url: "test.html",
    success: function(){
      alert("成功了!");
    },
    error:function(){
      alert("出错啦!");
    }
  });

新的写法:

//更简洁的链式写法
$.ajax("test.html")
  .done(function(){ alert("成功了!"); })
  .fail(function(){ alert("出错啦!"); });

而且deferred对象不仅仅是在ajax中使用,也可以在任何回调中进行使用。

比如:

var $dfd = $.Deferred();//创建一个deferred实例对象
var success = 0;//模拟成功失败状态
window.setTimeout(() => {
    if(success === 0){
        $dfd.resolve();
    }else{
         $dfd.reject();
    }
}, 100);
//两种写法:
$dfd.done(function(){
    alert('成功');
}).fail(function(){
    alert('失败');
});
//------另一种写法
$dfd.then(function(){
     alert('成功');
},function(){
     alert('失败');
});

同样的道理,使用deferred对象重写书写上面的案例:

<head>
    <meta charset="utf-8">
    <title>Index</title>
    <script src="https://code.jquery.com/jquery-3.3.1.min.js"></script>
</head>
<body>
    <div id="app"></div>
    <template id="template">
        <p></p>
    </template>
</body>
<script type="text/javascript">
    var app = $('#app');
    var p = $('#template')[0].content.querySelector('p');
    var p1 = p.cloneNode(true);
    var p2 = p.cloneNode(true);
    var p3 = p.cloneNode(true);

    var i = 0, html;
    var dfd1 = $.Deferred(),
        dfd2 = $.Deferred(),
        dfd3 = $.Deferred();
    window.setTimeout(() => {
        p1.innerHTML = '<p>aaaaaaaa</p>';
        dfd1.resolve();
    }, 100);
    window.setTimeout(() => {
        p2.innerHTML = '<p>bbbbbbbb</p>';
        dfd2.resolve();
    }, 200);
    window.setTimeout(() => {
        p3.innerHTML = '<p>cccccccc</p>';
        dfd3.resolve();
    }, 50);

    dfd1.done(function(){
        app.append(p1);
        return dfd2;
    }).done(function(){
        app.append(p2);
        return dfd3;
    }).done(function(){
        app.append(p3);
    });
</script>
</html>

另外:jquery还提供了一个同时监听多个回调的方法。$.when(),返回值为defferred对象。

$.when()接受多个deferred对象作为参数,当它们全部运行成功后,才调用resolved状态的回调函数,但只要其中有一个失败,就调用rejected状态的回调函数。它相当于将多个非同步操作,合并成一个。实质上,when方法为多个deferred对象,返回一个单一的promise对象。

$.when(
    $.ajax( "/main.php" ),
    $.ajax( "/modules.php" ),
    $.ajax( "/lists.php" )
).then(successFunc, failureFunc);
参考来源:

https://wangdoc.com/javascript/async/promise.html
https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Guide/Using_promises
http://javascript.ruanyifeng.com/jquery/deferred.html#toc0

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,772评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,458评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,610评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,640评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,657评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,590评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,962评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,631评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,870评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,611评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,704评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,386评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,969评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,944评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,179评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,742评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,440评论 2 342

推荐阅读更多精彩内容