jQuery.Deferred 源码分析
1 引子
观察者模式是我们日常开发中经常用的模式。这个模式由两个主要部分组成:发布者和观察者。通过观察者模式,实现发布者和观察者的解耦。
发布者主要负责发布内容,观察者主要负责监听发布者发布的内容,并作出相应的动作。和我们平时订阅期刊一样,cnki会维护一个订阅者列表,有期刊被发布出来时,cnki会将这些期刊推送给订阅者。从程序角度来说,订阅者就是一堆的方法,发布者的推送内容的动作就是依次调用订阅者列表中的方法(订阅者),而发布的内容就将以参数的形式提供给订阅者。

在jQuery中Deferred就是用来实现这一模式的。
我们首先来一个简单的订阅与发布示例:
//创建一个发布者
var subject=$.Deferred();
//创建两个订阅者
function subscriber1(content){
console.log(content);
}
function subscriber2(content){
console.log(content);
}
//订阅内容
subject.done(subscriber1,subscriber2);
//发布内容
subject.resolve('谢谢你的订阅');

下面我们来分析,$.Deferred的源码,在这个过程中,我们还会$.Deferred的各种使用情况。
2 构建Deferred
首先我们将Deferred源码做简化处理:

其中,tuples是一种数据结构。它是一种非常巧妙的设计,是把有共性的代码合并到一起,然后一次性进行处理。promise是deferred的一种简化形式(去掉了改变状态的接口)。Deferred是一个工厂类,返回的是内部创建好的deferred对象。
var tuples =[
// action, add listener, listener list, final state
//once 表示resolve和reject只要触发过一次,后面就不能再被触发了
["resolve","done", jQuery.Callbacks("once memory"),"resolved"],
["reject","fail", jQuery.Callbacks("once memory"),"rejected"],
["notify","progress", jQuery.Callbacks("memory")]
]
tuples定义3中状态的,即resolve,reject,notify。这三种状态分别对应了3中订阅接口(done,fail,progress)和3种订阅列表(3个Callbacks),resolve和reject有最终的状态(resolved和rejected)。下面就是为找这个数据结构,来构建带这3中状态的Deferred。
首先定义了初始状态:state = "pending"
然后构建promise对象:

在构建deferred对象:

构建deferred分为2步:
1. 构建3种状态对应的接口,订阅者列表,和推送内容接口。
构建订阅者列表,行,列表由一个Callbacks来管理,其实整个Deferred就是基于Callbacks来实现的,把它们的代码放在一起对比,会发现出奇的相似,构建状态字符串(分别为resolved、rejected或者undefined)。
$.each执行完会有3个列表生成:
doneList=$.Callbacks("once memory");
failList=$.Callbacks("once memory");
progressList=jQuery.Callbacks("memory");
定义订阅接口行,他们就是订阅列表(Callbacks)的add方法。add进去的订阅者会存储在Callbacks的list数组中。
3中订阅接口分别为:
promise.done=doneList.add;
promise.fail=failList.add;
promise.progress=progressList.add
为resolved和rejected两种状态的列表添加预设的3个回调函数(3437~3444)。第一个回调函数用于处理Deferred的状态(已成功或已失败),第二个回调函数用于让另一个列表失效,这是因为当发布者发布内容的动作已经成功了,它就不可能再触发失败的状态,因此让失败列表失效,防止给正在订阅了失败状态的函数被执行。反之亦然。然后锁定notify列表。锁定notify只是让其的progressList的fire方法(即后面不能使用notify和notifyWith再进行推送内容了)不能被调用了。但是,我们还可以给notify状态继续添加订阅者,这些订阅者会立即被执行。
这里有个小技巧:i^1,^是按位运算符

没什么大用,但是jQuery就是要连这点性能也要节省出来。
至此,doneList的list中有3个回调函数[changeState, failList.disable, progressList.lock],failList的list中同样有3个回调函数[changeState, doneList.disable, progressList.lock]
构建推送内容接口(3447~3451),三种接口名字分别为
deferred.resolve(成功),deferred.reject(失败),deferred.notify(正在执行),这三个接口在内部其实又去调用这三个接口对应的带参数版本(deferred.resolveWith,deferred.rejectWith,deferred.notifyWith),而三个接口又是订阅列表(Callbacks)的fireWith方法。由此可见,要能非常明白的看懂Deferred的源码,必须对Callbacks的源码非常熟悉,至少要能熟练运用Callbacks。
2. 将deferred的精简版promise中的方法和属性或从到自己身上。()这里的promise也是一种非常巧妙的处理。
构造结束后我们就会得到如下图所示的Deferred对象:

至此,Deferred对象构建完成。
比较发现,promise比deferred少了6个内容推送功能。没有了这几个功能deferred的状态也就不会被改变。
3 Deferred的使用
3.1 done,resolve; fail,reject; progress,notify
setTimeout(function () {
console.log('耗时操作……');
def.resolve('本次推送的内容……');
},);
def.done(function () {
console.log(arguments[]);
});

这个过程非常简单,只是对Callbacks的简单包装。
var def=$.Deferred();
setTimeout(function () {
new Error('不好意思,出错了……');
def.reject('台风影响,本次推送出错……');
},);
def.done(function () {
console.log(arguments[]);
}).fail(function () {
console.log(arguments[]);
});
这里的原理和是那个面的一模一样。
var def=$.Deferred();
setTimeout(function() {
console.log('正在执行……');
def.notify('正在执行……');
},);
def.done(function () {
console.log(arguments[]);
}).fail(function () {
console.log(arguments[]);
}).progress(function() {
console.log(arguments[]);
});
3.2 then()
var def=$.Deferred();
setTimeout(function () {
console.log('正在执行……');
def.notify('正在执行……');
},);
def.then(function () {
console.log(arguments[]);
},function () {
console.log(arguments[]);
},function () {
console.log(arguments[]);
});

then方法返回一个新的的Deferred对象的精简版promise。在创建该Deferred是传递进去了一个function参数。我们发现,在构建Deferred的工厂方法就有一个func参数,它内部的处理是这样的:

以工厂方法创建的deferred为上下文,以该deferred为参数来调用这个方法。函数执行完后,工厂方法将deferred返回。
然后我们来看这个func具体是怎么设计的,这里的设计的非常之复杂。这里最重要的概念就是作用域链,如果对这些基础知识不熟悉的话,理解起来真的很费劲……

以上面实例代码为例,来分析then的源码:

分析源码不仅可以回顾JavaScript知识,还可以借鉴别人的设计思路,代码书写方式,对提高自身的修养是很有帮助的。
下一章分析jQuery.when的实现……
jQuery.Deferred 源码分析的更多相关文章
- jQuery.attributes源码分析(attr/prop/val/class)
回顾 有了之前的几篇对于jQuery.attributes相关的研究,是时候分析jQuery.attr的源码了 Javascript中的attribute和property分析 attribute和p ...
- 移动web app开发必备 - Deferred 源码分析
姊妹篇 移动web app开发必备 - 异步队列 Deferred 在分析Deferred之前我觉得还是有必要把老套的设计模式给搬出来,便于理解源码! 观察者模式 观察者模式( 又叫发布者-订阅者模 ...
- jQuery.queue源码分析
作者:禅楼望月(http://www.cnblogs.com/yaoyinglong ) 队列是一种特殊的线性表,它的特殊之处在于他只允许在头部进行删除,在尾部进行插入.常用来表示先进先出的操作(FI ...
- jQuery.buildFragment源码分析以及在构造jQuery对象的作用
这个方法在jQuery源码中比较靠后的位置出现,主要用于两处.1是构造jQuery对象的时候使用 2.是为DOM操作提供底层支持,这也就是为什么先学习它的原因.之前的随笔已经分析过jQuery的构造函 ...
- 深入分析,理解jQuery.Deferred源码
前言: 如果你对jQuery.Callback回调对象不了解,或者只掌握其方法,但是没有通过阅读源码理解,可以先阅读 前一章jQuery.Callbacks源码解读二,因为只有完全理解jQuery.C ...
- jQuery.access源码分析
基本理解 jQuery.attr是jQuery.attr,jQuery.prop,jQuery.css提供底层支持,jQuery里一个比较有特色的地方就是函数的重载, 比如attr,有如下几种重载 $ ...
- jQuery 2.1.4版本的源码分析
jQuery 2.1.4版本的源码分析 jquery中获取元素的源码分析 jQuery.each({// 获取当前元素的父级元素 parent: function(elem) { var parent ...
- jQuery.Callbacks 源码解读二
一.参数标记 /* * once: 确保回调列表仅只fire一次 * unique: 在执行add操作中,确保回调列表中不存在重复的回调 * stopOnFalse: 当执行回调返回值为false,则 ...
- jQuery 2.0.3 源码分析 Deferred(最细的实现剖析,带图)
Deferred的概念请看第一篇 http://www.cnblogs.com/aaronjs/p/3348569.html ******************构建Deferred对象时候的流程图* ...
随机推荐
- iOS应用内跳转系统设置相关界面的方法
在iOS开发中,有时会有跳转系统设置界面的需求,例如提示用户打开蓝牙或者WIFI,提醒用户打开推送或者位置权限等.在iOS6之后,第三方应用需要跳转系统设置界面,需要在URL type中添加一个pre ...
- ehcache基本使用
maven <dependency> <groupId>net.sf.ehcache</groupId> <artifactId>ehcache< ...
- asp.net identity 2.2.0 中角色启用和基本使用(二)
建立模型 第一步:在Models文件夹上点右键 >添加>类 类的名称自定,我用AdminViewModels命名的 因为是讲基本使用,我这里不做任何扩展. 第二步:添加如下命名空间 ...
- Backbone源码解析(四):View(视图)模块
View视图故名思义,它控制的是界面.我们可以把一个大的网页分成很多部分的视图,按照backbone的架构,每一个视图对应都是一个对象,我们可以通过元素的钩子(id或者class或者其他选择器)把它们 ...
- ddms(基于 Express 的表单管理系统)源码学习
ddms是基于express的一个表单管理系统,今天抽时间看了下它的代码,其实算不上源码学习,只是对它其中一些小的开发技巧做一些记录,希望以后在项目开发中能够实践下. 数据层封装 模块只对外暴露mod ...
- java 多线程(ThreadPoolExecutor (补充))
import java.util.concurrent.ArrayBlockingQueue; import java.util.concurrent.ThreadPoolExecutor; impo ...
- C#Light 和 uLua的对比第二弹
上次的对比大家还有印象否,C#Light和ulua对比各有胜负 今天我们加入一个去反射优化,这是uLua没办法实现的优化,我们也就只能不要脸的胜之不武了 以原生执行同一测试时间为X1,数字越小的越快 ...
- Android Fragment add/replace以及backstack
无论Fragment以何种方式加入,都不会影响backstack,backstack由addToBackStack函数决定,只有调用了这个函数,才会将Fragment加入返回栈.这个说法其实不太准确, ...
- NanoProfiler - 适合生产环境的性能监控类库 之 实践ELK篇
上期回顾 上一期:NanoProfiler - 适合生产环境的性能监控类库 之 大数据篇 上次介绍了NanoProfiler的大数据分析理念,一晃已经时隔一年多了,真是罪过! 有朋友问到何时开源的问题 ...
- 容器使用的12条军规——《Effective+STL中文版》试读
容器使用的12条军规——<Effective+STL中文版>试读 还 记的自己早年在学校学习c++的时候,老师根本就没有讲STL,导致了自己后来跟人说 起会C++的时候总是被鄙视, ...