一, 前言                            

深入学习Promise的朋友应该都看过<深入理解Promise五部曲>这一系列的文章, 以解除回调地狱之外的观点来剖析Promise更多的内涵,确实十分精彩.

Part 1: The Sync Problem(译文:http://segmentfault.com/blog/kk_470661/1190000000586666)

Part 2: The Inversion Problem(译文:http://segmentfault.com/blog/kk_470661/1190000000591382)

Part 3: The Trust Problem(译文:http://segmentfault.com/blog/kk_470661/1190000000593885)

Part 4: The Extension Problem(译文:http://segmentfault.com/blog/kk_470661/1190000000600268)

Part 5: The LEGO Problem(译文:http://segmentfault.com/blog/kk_470661/1190000000611040)

NPO(Native Promise Only)是原文作者polyfill的ES6 Promise, 本文为拜读文章及源码后的笔记,以便日后查阅.

NPO@github

二, 整体脉络                          

对于Promise实现而言, 主要的主体类型就两个-----Promise和Thenable. NPO中通过MakeDef构建Promise的内部状态结构体def, 并且通过def.chain存储Promise子节点P2-1,P2-2到P2-n, 从而形成一颗Promise树. 而Thenable的内部状态结构体def_wrapper则由MakeDefWrapper构建而成.

Promise树的结构并不稳定, 实际上每个Promise节点仅与状态为pending的子节点关联, 一旦子节点状态发生变化则断开关联.(该部分在 notify() 中实现)

{Promise} then(success, failure) , 将success和failure事件处理函数与新生成的Promise子节点绑定, 但订阅的是Promise父节点的状态变化事件.

另外NPO中通过构建一个异步执行请求队列(scheduling_queue),来收集异步执行请求然后對请求作同一处理,并通过门闩(cycle)来防止重复执行异步请求处理操作.

三, 源码详解                            

先看看Promise构造函数, 规定仅能通过new方式来构建Promise实例.

function Promise(executor) {
if (typeof executor != "function") {
throw TypeError("Not a function");
} if (this.__NPO__ !== ) {
throw TypeError("Not a promise");
} // instance shadowing the inherited "brand"
// to signal an already "initialized" promise
this.__NPO__ = ; // 内部结构体
var def = new MakeDef(this); this["then"] = function then(success,failure) {
var o = {
success: typeof success == "function" ? success : true,
failure: typeof failure == "function" ? failure : false
};
// Note: `then(..)` itself can be borrowed to be used against
// a different promise constructor for making the chained promise,
// by substituting a different `this` binding.
o.promise = new this.constructor(function extractChain(resolve,reject) {
if (typeof resolve != "function" || typeof reject != "function") {
throw TypeError("Not a function");
} o.resolve = resolve;
o.reject = reject;
});
// 构建Promise树
def.chain.push(o);
             // 当前Promise节点状态不为pending时,发起异步执行请求事件处理函数
if (def.state !== ) {
schedule(notify,def);
} return o.promise;
};
this["catch"] = function $catch$(failure) {
return this.then(void ,failure);
}; try {
// 调用工厂方法
executor.call(
void ,
function publicResolve(msg){
resolve.call(def,msg);
},
function publicReject(msg) {
reject.call(def,msg);
}
);
}
catch (err) {
reject.call(def,err);
}
}

Promise的状态变化放在resolve和reject函数中

 function resolve(msg) {
var _then, def_wrapper, self = this; // already triggered?
if (self.triggered) { return; } self.triggered = true; // unwrap
if (self.def) {
self = self.def;
} try {
if (_then = isThenable(msg)) {
          // 构造Thenable的内部状态结构体
def_wrapper = new MakeDefWrapper(self);
_then.call(msg,
function $resolve$(){ resolve.apply(def_wrapper,arguments); },
function $reject$(){ reject.apply(def_wrapper,arguments); }
);
}
else {
self.msg = msg;
self.state = ;
if (self.chain.length > ) {
schedule(notify,self);
}
}
}
catch (err) {
reject.call(def_wrapper || (new MakeDefWrapper(self)),err);
}
}
function reject(msg) {
var self = this; // already triggered?
if (self.triggered) { return; } self.triggered = true; // unwrap
if (self.def) {
self = self.def;
} self.msg = msg;
self.state = ;
if (self.chain.length > ) {
schedule(notify,self);
}
}

下面看一下我觉得最亮眼的地方异步执行请求队列, 主要由以下几个部分组成

1. notify, 遍历def.chain中的所有Promise子节点, 最后由于所有Promise子节的状态均变为fulfilled或rejected因此清空def.chain.

2. notifyIsolated, 被notify所调用, 用于单独调用绑定在每个Promise子节点的success或failure事件处理函数, 并修改Promse子节点的状态.

3. scheduling_queue, 存放异步执行请求(以链表实现, 對队列首尾操作性能比数组高).

4. schedule, 向异步执行请求队列添加元素, 并发起异步请求处理操作.

上述的1和2两点将作为异步执行请求被存放在3中.代码中各部分则通过4来對队列和异步执行请求作操作.

        function notify() {
for (var i=; i<this.chain.length; i++) {
notifyIsolated(
this,
(this.state === ) ? this.chain[i].success : this.chain[i].failure,
this.chain[i]
);
}
this.chain.length = ;
} // NOTE: This is a separate function to isolate
// the `try..catch` so that other code can be
// optimized better
function notifyIsolated(self,cb,chain) {
var ret, _then;
try {
if (cb === false) {
chain.reject(self.msg);
}
else {
if (cb === true) {
ret = self.msg;
}
else {
ret = cb.call(void ,self.msg);
} if (ret === chain.promise) {
chain.reject(TypeError("Promise-chain cycle"));
}
else if (_then = isThenable(ret)) {
_then.call(ret,chain.resolve,chain.reject);
}
else {
chain.resolve(ret);
}
}
}
catch (err) {
chain.reject(err);
}
}
scheduling_queue = (function Queue() {
var first // 指向队首元素
, last // 指向队尾元素
, item; function Item(fn,self) {
this.fn = fn;
this.self = self;
this.next = void ;
} return {
// 元素入队
add: function add(fn,self) {
item = new Item(fn,self);
if (last) {
last.next = item;
}
else {
first = item;
}
last = item;
item = void ;
},
// 清空队列
drain: function drain() {
var f = first;
first = last = cycle = void ; // 从队首元素开始遍历所有队列元素
while (f) {
f.fn.call(f.self);
f = f.next;
}
}
};
})(); // 安排执行状态变化事件的处理函数
function schedule(fn,self) {
scheduling_queue.add(fn,self);
// 防止重复发起异步执行请求
if (!cycle) {
cycle = timer(scheduling_queue.drain);
}
}

四, 总结                            

尊重原创,转载请注明来自: http://www.cnblogs.com/fsjohnhuang/p/4293499.html ^_^肥仔John

JS魔法堂: Native Promise Only源码剖析的更多相关文章

  1. JS魔法堂:jsDeferred源码剖析

    一.前言 最近在研究Promises/A+规范及实现,而Promise/A+规范的制定则很大程度地参考了由日本geek cho45发起的jsDeferred项目(<JavaScript框架设计& ...

  2. JS魔法堂:剖析源码理解Promises/A规范

    一.前言 Promises/A是由CommonJS组织制定的异步模式编程规范,有不少库已根据该规范及后来经改进的Promises/A+规范提供了实现 如Q, Bluebird, when, rsvp. ...

  3. JS魔法堂:mmDeferred源码剖析

    一.前言 avalon.js的影响力愈发强劲,而作为子模块之一的mmDeferred必然成为异步调用模式学习之旅的又一站呢!本文将记录我对mmDeferred的认识,若有纰漏请各位指正,谢谢.项目请见 ...

  4. JS魔法堂:IMG元素加载行为详解

    一.前言 在<JS魔法堂:jsDeferred源码剖析>中我们了解到img元素加载失败可以作为函数异步执行的优化方案,本文打算对img元素的加载行为进行更深入的探讨. 二.资源加载的相关属 ...

  5. JS魔法堂:属性、特性,傻傻分不清楚

    一.前言 或许你和我一样都曾经被下面的代码所困扰 var el = document.getElementById('dummy'); el.hello = "test"; con ...

  6. JS魔法堂:判断节点位置关系

    一.前言 在polyfill querySelectorAll 和写弹出窗时都需要判断两个节点间的位置关系,通过jQuery我们可以轻松搞定,但原生JS呢?下面我将整理各种判断方法,以供日后查阅. 二 ...

  7. JS魔法堂:LINK元素深入详解

    一.前言 我们一般使用方式为 <link type="text/css" rel="stylesheet" href="text.css&quo ...

  8. JS魔法堂:追忆那些原始的选择器

    一.前言                                                                                                 ...

  9. Promise的源码实现(完美符合Promise/A+规范)

    Promise是前端面试中的高频问题,我作为面试官的时候,问Promise的概率超过90%,据我所知,大多数公司,都会问一些关于Promise的问题.如果你能根据PromiseA+的规范,写出符合规范 ...

随机推荐

  1. Mroonga 3.0.8 发布,MySQL 存储引擎

    Mroonga 3.0.8 支持 REPAIR TABLE 支持损坏的 groonga 数据库. Mroonga 是一个 MySQL 存储引擎,基于 Groonga,提供完整的全文搜索引擎.

  2. android oauth 微博客户端 架构一

    最近研究oauth协议,为了进一步 的巩固自己的学习成果,顾完成了android的新浪客户端.他的架构如下: UI层微博中的各个窗体  就是所谓的各个activitylogic层程序的核心控制调度模块 ...

  3. Web服务器常用设置

    1.Tomcat浏览目录 找到安装目录下的文件/conf/web.xml,  找到以下配置节,将parame-value设置为true即可 <init-param>             ...

  4. 开始VS 2012中LightSwitch系列的第3部分:我该选择哪一个屏幕模板

    [原文发表地址]  Beginning LightSwitch in VS 2012 Part 3: Screen Templates, Which One Do I Choose? [原文发表时间] ...

  5. 异步CTP(Async CTP)为什么那样工作?

    对异步CTP感兴趣有很多原因.异步CTP使异步编程比以前更加容易了.它虽然没有Rx强大,但是更容易学.异步CTP介绍了两个新的关键字,async和await.异步方法(或Lambda表达式)必须返回v ...

  6. 人人都是 DBA(XV)锁信息收集脚本汇编

    什么?有个 SQL 执行了 8 秒! 哪里出了问题?臣妾不知道啊,得找 DBA 啊. DBA 人呢?离职了!!擦!!! 程序员在无处寻求帮助时,就得想办法自救,努力让自己变成 "伪 DBA& ...

  7. Linux cat命令

    200 ? "200px" : this.width)!important;} --> 介绍 cat命令经常会用来查看一个文件的内容,并且结合它本身的一些参数经常可以用来做一 ...

  8. C++虚函数表

    大家知道虚函数是通过一张虚函数表来实现的.在这个表中,主要是一个类的虚函数的地址表,这张表解决了继承.覆盖的问题,其内容真是反应实际的函数.这样,在有虚函数的类的实例中,这个表分配在了这个实例的内存中 ...

  9. springmvc下实现登录验证码功能

    总体思路,简单讲,就是后台生成图片同时将图片信息保存在session,前端显示图片,输入验证码信息后提交表单到后台,取出存放在session里的验证码信息,与表单提交的验证码信息核对. 点击验证码图片 ...

  10. PLoP(Pattern Languages of Programs,程序设计的模式语言)

    2014/8/1 12:24:21潘加宇 http://www.umlchina.com/News/Content/340.htmPloP大会2014即将举行 PLoP(Pattern Languag ...