jQuery源码分析系列(38) : 队列操作
Queue队列,如同data数据缓存与Deferred异步模型一样,都是jQuery库的内部实现的基础设施
Queue队列是animate动画依赖的基础设施,整个jQuery中队列仅供给动画使用
Queue队列
队列是一种特殊的线性表,只允许在表的前端(队头)进行删除操作(出队),在表的后端(队尾)进行插入操作(入队)。队列的特点是先进先出(FIFO-first in first out),即最先插入的元素最先被删除。
为什么要引入队列?
我们知道代码的执行流有异步与同步之分,例如
var a = 1; setTimeout(function(){
a = 2;
},0) alert(a) //
我们一直习惯于“线性”地编写代码逻辑,但是在JavaScript编程几乎总是伴随着异步操作:
setTimeout,CSS3 Transition/Animation,ajax,dom的绘制,postmessage,Web Database等等,大量异步操作所带来的回调函数,会把我们的算法分解地支离破碎
之前我们说过对于异步+回调的模式,怎么“拉平”异步操作,使之跟同步一样,因为异步操作进行流程控制的时候无非避免的要嵌套大量的回调逻辑,所以就会出现promises约定了
那么jQuery引入队列其实从一个角度上可以认为:允许一系列函数被异步地调用而不会阻塞程序
$("#Aaron").slideUp().fadeIn()
这是jQuery的一组动画链式序列,它的内部其实就是一组队列Queue,所以队列和Deferred地位类似, 是一个内部使用的基础设施,当slideUp运行时,fadeIn被放到fx队列中,当slideUp完成后,从队列中被取出运行。queue函数允许 直接操作这个链式调用的行为。同时,queue可以指定队列名称获得其他能力,而不局限于fx队列
jQuery提供了2组队列操作的API:
- jQuery.queue/dequeue
- jQuery.fn.queue/dequeue
但是不同与普通队列定义的是:jQuery.queue和jQuery.fn.queue不仅执行出队操作,返回队头元素,还会自动执行返回的队头元素
fn是扩展在原型上的高级API是提供给实例使用的,.queue/.dequeue, 其内部是调用的$.queue,$.dequeue静态的底层方法实现入列与出列
$.queue : 显示或操作匹配的元素上已经执行的函数列队
这个方法有两个作用,它既是setter,又是getter。第一个参数elem是DOM元素,第二个参数type是字符串,第三个参数data可以是function或数组。
var body = $('body');
function cb1() {alert(1)}
function cb2() {alert(2)} //set
$.queue(body, 'aa', cb1); // 第三个参数为function
$.queue(body, 'aa', cb2); //get
$.queue(body, 'aa') //[function ,function]
这个方法有点类型get有点类似队列的push操作,jQuery的方法的接口重载是非常严重的,经常同一个接口即是set也是get,不管符不符合基本原则,但是它却很实用
无非就是把数据给缓存起来,为什么载体是一个jQuery对象,因为保存数据的手段是通过data数据缓存实现的
data_priv = new Data();
queue: function(elem, type, data) {
var queue;
if (elem) {
type = (type || "fx") + "queue";
queue = data_priv.get(elem, type);
// Speed up dequeue by getting out quickly if this is just a lookup
if (data) {
if (!queue || jQuery.isArray(data)) {
queue = data_priv.access(elem, type, jQuery.makeArray(data));
} else {
queue.push(data);
}
}
return queue || [];
}
},
data与jQuery对象之间是通过uuid建立了一个无耦合的映射关系,具体可以翻阅之前的关于“数据缓存”
源码有一个默认处理
type = (type || "fx") + "queue"
可见是专职供fx动画队列处理的
$.dequeue : 匹配的元素上执行队列中的下一个函数
var body = $('body');
function cb1() {console.log(11111)}
function cb2() {console.log(22222)} //set
$.queue(body, 'aa', cb1); // 第三个参数为function
$.queue(body, 'aa', cb2); $.dequeue(body, 'aa') //
$.dequeue(body, 'aa') //
出列就有点类似shift的操作,但是不同的是还会执行这个cb1与cb2
将回调函数出列执行,每调用一次仅出列一个,因此当回调有N个时,需要调用$.dequeue方法N次元素才全部出列
来看看源码:
var queue = jQuery.queue(elem, type),
startLength = queue.length,
fn = queue.shift(),
hooks = jQuery._queueHooks(elem, type),
next = function() {
jQuery.dequeue(elem, type);
};
知道原理了, 这个就很简单了,通过queue的get取出队列的所有数据,判断一下长度,然后截取出第一个,然后做好一个预处理生成下一个的next
这里有一个hooks?
仔细分析下这个内部queueHooks
_queueHooks: function(elem, type) {
var key = type + "queueHooks";
return data_priv.get(elem, key) || data_priv.access(elem, key, {
empty: jQuery.Callbacks("once memory").add(function() {
data_priv.remove(elem, [type + "queue", key]);
})
});
}
我们说了dequeue不仅是取出来还需要执行,在执行的时候把next与hooks传递给外部的回调,
这就是js的逻辑上的很绕的地方,在内部可以传递一个引用出去,又能提供外部调用或者执行
fn.call(elem, next, hooks)
因为传递了next,所以我们的代码可以这样改
var body = $('body');
function cb1(next,hoost) {
console.log(11111)
next() //执行了cb2 //22222
} function cb2() {
console.log(22222)
} //set
$.queue(body, 'aa', cb1); // 第三个参数为function
$.queue(body, 'aa', cb2); $.dequeue(body, 'aa')
next内部仍然调用$.dequeue,这样可以接着执行队列中的下一个callback
$.dequeue里的hooks是当队列里所有的callback都执行完后(此时startLength为0)进行最后的一个清理工作
if ( !startLength && hooks ) {
hooks.empty.fire();
}
钩子其实就是jQuery.Callbacks对象,可以实现一个收集器的功能,至于在什么情况下时候,之后动画中开始分析
所以队列的本质是利用Array的push和shift来完成先进先出(First In First Out),但是这个方法的缺点也很明显,无法单独做一个独立的模块处理,因为它必须要跟jQuery对象吻合,而且对传递的数据只能是函数
jQuery源码分析系列(38) : 队列操作的更多相关文章
- jQuery源码分析系列
声明:本文为原创文章,如需转载,请注明来源并保留原文链接Aaron,谢谢! 版本截止到2013.8.24 jQuery官方发布最新的的2.0.3为准 附上每一章的源码注释分析 :https://git ...
- [转]jQuery源码分析系列
文章转自:jQuery源码分析系列-Aaron 版本截止到2013.8.24 jQuery官方发布最新的的2.0.3为准 附上每一章的源码注释分析 :https://github.com/JsAaro ...
- jQuery源码分析系列(转载来源Aaron.)
声明:非本文原创文章,转载来源原文链接Aaron. 版本截止到2013.8.24 jQuery官方发布最新的的2.0.3为准 附上每一章的源码注释分析 :https://github.com/JsAa ...
- jQuery源码分析系列——来自Aaron
jQuery源码分析系列——来自Aaron 转载地址:http://www.cnblogs.com/aaronjs/p/3279314.html 版本截止到2013.8.24 jQuery官方发布最新 ...
- jQuery源码分析(九) 异步队列模块 Deferred 详解
deferred对象就是jQuery的回调函数解决方案,它解决了如何处理耗时操作的问题,比如一些Ajax操作,动画操作等.(P.s:紧跟上一节:https://www.cnblogs.com/grea ...
- jQuery 源码分析(二十一) DOM操作模块 删除元素 详解
本节说一下DOM操作模块里的删除元素模块,该模块用于删除DOM里的某个节点,也可以理解为将该节点从DOM树中卸载掉,如果该节点有绑定事件,我们可以选择保留或删除这些事件,删除元素的接口有如下三个: e ...
- jQuery 源码分析(二十) DOM操作模块 插入元素 详解
jQuery的DOM操作模块封装了DOM模型的insertBefore().appendChild().removeChild().cloneNode().replaceChild()等原生方法.分为 ...
- jQuery 源码分析(十二) 数据操作模块 html特性 详解
jQuery的属性操作模块总共有4个部分,本篇说一下第1个部分:HTML特性部分,html特性部分是对原生方法getAttribute()和setAttribute()的封装,用于修改DOM元素的特性 ...
- jQuery 源码分析(十五) 数据操作模块 val详解
jQuery的属性操作模块总共有4个部分,本篇说一下最后一个部分:val值的操作,也是属性操作里最简单的吧,只有一个API,如下: val(vlaue) ;获取匹配元素集合中第一个元素的 ...
随机推荐
- 求50-100内的素数(java)
实现代码: public class sushu { public static void main(String[] args) { for(int i=50 ; i<=100; i++){ ...
- SOAPUI使用教程-MockService脚本概述
虽然静态MockOperation和MockResponse模型可以给你一定的灵活性,更多的动态行为添加到您的服务让您更模拟复杂的服务功能是很有用的.例如,你可能想: 从请求到响应传输值,修改它们的值 ...
- XV Open Cup named after E.V. Pankratiev. GP of Tatarstan
A. Survival Route 留坑. B. Dispersed parentheses $f[i][j][k]$表示长度为$i$,未匹配的左括号数为$j$,最多的未匹配左括号数为$k$的方案数. ...
- 关于PHP语言
------php语言与JavaScript的使用 方法是相似 <script type="text/javascript"> </script>--js与 ...
- Agile
I think Agile development methodologies is something we get from our practice. It can be just acknow ...
- 仿【Emmet】转【HTML】功能
一.定义正则表达式: var whitespace = "[\\x20\\t\\r\\n\\f]*", nvarcharEncoding = whitespace + " ...
- Cannot run gnome extension in browser
Error Message: We cannot detect a running copy of GNOME on this system, so some parts of the interfa ...
- 【转】最大流EK算法
转自:http://www.cnblogs.com/kuangbin/archive/2011/07/26/2117636.html 图-1 如图-1所示,在这个运输网络中,源点S和汇点T分别是1,7 ...
- jquery如何获取第一个或最后一个子元素?
通过children方法,children("input:first-child") 1 2 $(this).children("input:first-child&qu ...
- ICollection
ICollection 接口是 System.Collections 命名空间中类的基接口.ICollection 接口扩展 IEnumerable:IDictionary 和 IList 则是扩展 ...