jquery源码解析:expando,holdReady,ready详解
jQuery的工具方法,其实就是静态方法,源码里面就是通过extend方法,把这些工具方法添加给jQuery构造函数的。
jQuery.extend({ //当只有一个对象时,就把这个对象中的属性和方法扩展到this对象中,这里的this指向jQuery
expando: "jQuery" + ( core_version + Math.random() ).replace( /\D/g, "" ),
//唯一性,core_version 为jQuery的版本号。\D的意思是:不是数字的,就选中。因此expando就是jQuery+一段整数。数据缓存,ajax,事件机制都用到了这个。
noConflict: function( deep ) {
//处理冲突,因为有些库可能会用到$甚至jQuery,这时就可以调用chaojidan = $.onConflict(true);这时chaoijdan就可以当做jQuery了,并且其他库可以使用$标识了
if ( window.$ === jQuery ) {
window.$ = _$;
}
if ( deep && window.jQuery === jQuery ) {
window.jQuery = _jQuery;
}
return jQuery;
},
isReady: false,
readyWait: 1,
holdReady: function( hold ) {
if ( hold ) {
jQuery.readyWait++;
} else {
jQuery.ready( true );
}
},
ready: function( wait ) {
if ( wait === true ? --jQuery.readyWait : jQuery.isReady ) {
return;
}
jQuery.isReady = true;
if ( wait !== true && --jQuery.readyWait > 0 ) {
return;
}
readyList.resolveWith( document, [ jQuery ] );
if ( jQuery.fn.trigger ) {
jQuery( document ).trigger("ready").off("ready");
}
},
//以上几个属性和方法其实就是解决DOM加载问题的,大家都知道onload是等DOM树建造完成,并且资源加载完成才会触发。但是我们只需要DOM树构造完成,就执行绑定事件,所以有一个DOMContentLoaded事件。但是低版本浏览器不支持,所以需要兼容,jQuery就写了兼容方法。所以在这里介绍一下它是怎么处理的。
$(function(){}) -> $(document).ready(function(){}) -> $().ready() -> jQuery.ready.promise().done(fn),
jQuery.ready.promise = function( obj ) {
if ( !readyList ) { //第一次调用时,为空,所以进入。之后就直接使用这个对象
readyList = jQuery.Deferred(); //新建一个延迟对象
if ( document.readyState === "complete" ) {
//DOM已经加载结束了,这时就可以直接调用jQuery.ready,不用再绑定事件了
setTimeout( jQuery.ready );
//DOM快加载好的时候,IE的readyState有可能就是complete了,这时调用jQuery.ready会出问题,因此延迟调用,确保IE下也没问题。
} else {
document.addEventListener( "DOMContentLoaded", completed, false );
window.addEventListener( "load", completed, false );
//火狐浏览器可能会缓存onload事件,这时onload会先于DOMContentLoaded触发。写两个,更保险
}
}
return readyList.promise( obj ); //延迟对象的状态有几种,在外面可以被修改,但是promise不能修改
};
completed = function() {
document.removeEventListener( "DOMContentLoaded", completed, false ); //只调用一次completed 方法
window.removeEventListener( "load", completed, false );
jQuery.ready();
};
//jQuery.ready.promise().done(fn) (延迟对象调用done方法,把fn方法加入延迟对象,当触发延迟对象的resolveWith时,就会执行fn方法) -> jQuery.ready() -> readyList.resolveWith( document, [ jQuery ] )
ready: function( wait ) {
if ( wait === true ? --jQuery.readyWait : jQuery.isReady ) {
return;
}
jQuery.isReady = true; //只触发一次
if ( wait !== true && --jQuery.readyWait > 0 ) { //DOM的ready正常触发时,也要检查是否需要等待
return;
}
readyList.resolveWith( document, [ jQuery ] );
//第一个参数是指向this(回调方法的执行上下文),第二个参数是传给回调方法的参数。这个回调方法,就是通过延迟对象的done方法加进来的fn.因此,resolveWith( document, [ jQuery ] );相当于fn(jQuery),在document中执行。
if ( jQuery.fn.trigger ) { //这里处理jQuery第三种DOM加载完成执行js的方式。
jQuery( document ).trigger("ready").off("ready"); //触发ready事件后,马上移除掉,只触发一次。
//$(document).on("ready",function(){}),通过document绑定ready事件来实现。
}
},
holdReady: function( hold ) { //阻止Dom加载完成执行function,$.holdReady(true),
if ( hold ) {
jQuery.readyWait++; //要加载多个文件时(a.js,b.js),就要每次都++。
} else {
jQuery.ready( true ); //$.holdReady(false),恢复执行,减少一次jQuery.readyWait,直到它为0.
}
},
这个方法有什么用呢?举个例子:$.getScript("a.js",function(){}); $(function(){ a.say(); });这个代码的意思是加载完a.js,然后再调用a的say 方法。但是a.js需要下载,所以执行a.say会报错。这时,可以先$.holdReady(true),这样$(function(){ a.say(); }中的a.say(); 就不会执行了。直到我在$.getScript("a.js",function(){$.holdReady(false)});才可以执行。而这时,a.js加载完成,并执行结束后,调用回调方法,执行$.holdReady(false),恢复执行。
isFunction: function( obj ) { //在低版本IE浏览器下typeof alert 返回object,而不是function
return jQuery.type(obj) === "function";
},
isArray: Array.isArray, //不兼容IE8以及以下版本
isWindow: function( obj ) {
return obj != null && obj === obj.window;
//false == null -> false ,true == null -> false。只有null和undefined才等于null,而只有这两个变量没有属性。其他的像字符串有包装对象,可以有属性。window有两种意思:(1)全局对象,(2)浏览器窗口。而只有window才有window属性,因此只有window才会返回true.window.window表示全局对象中的浏览器窗口。
},
isNumeric: function( obj ) { //typeof NaN -> number
return !isNaN( parseFloat(obj) ) && isFinite( obj ); //Number.MAX_VALUE最大值,加1就会是无限大,就不是number类型
//先判断参数是否能转成数字,不可以就是NaN,那么isNaN就会返回true。如果是数字,但是必须是有限数,不然也不是number类型。
},
...
})
加油!
jquery源码解析:expando,holdReady,ready详解的更多相关文章
- jQuery 源码解析(三) pushStack方法 详解
该函数用于创建一个新的jQuery对象,然后将一个DOM元素集合加入到jQuery栈中,最后返回该jQuery对象,有三个参数,如下: elems Array类型 将要压入 jQuery 栈的数组元素 ...
- guava-retrying 源码解析(停止策略详解)
一.停止策略相关类 1.停止策略接口:StopStrategy接口,只有一个抽象方法 // 是否应该停止重试.不同的停止策略有不同的实现.boolean shouldStop(Attempt fail ...
- guava-retrying 源码解析(等待策略详解)
一.等待策略相关类: 1.等待策略接口:WaitStrategy接口 该接口只有一个方法,就是返回尝试失败之后,下一次尝试之前的等待时间.long computeSleepTime(Attempt f ...
- vuex 源码解析(三) getter属性详解
有时候我们需要从store中的state中派生出一些状态,例如: <div id="app"> <p>{{reverseMessage}}</p> ...
- guava-retrying 源码解析(阻塞策略详解)
这是一种策略,用于决定重试者应如何在重试尝试之间进行阻止.通常这只是一个thread.sleep(),但是如果需要的话,实现可能更复杂. 一.阻塞策略相关的类或接口 1.阻塞策略接口:BlockStr ...
- JQuery源码解析(一)
写在前面:本<JQuery源码解析>系列是基于一些前辈们的文章进行进一步的分析.细化.修改而写出来的,在这边感谢那些慷慨提供科普文档的技术大拿们. 要查阅JQ的源文件请下载开发版的JQ.j ...
- jQuery 源码解析二:jQuery.fn.extend=jQuery.extend 方法探究
终于动笔开始 jQuery 源码解析第二篇,写文章还真是有难度,要把自已懂的表述清楚,要让别人听懂真的不是一见易事. 在 jQuery 源码解析一:jQuery 类库整体架构设计解析 一文,大致描述了 ...
- jquery源码解析:代码结构分析
本系列是针对jquery2.0.3版本进行的讲解.此版本不支持IE8及以下版本. (function(){ (21, 94) 定义了一些变量和函数, jQuery = function() ...
- jquery源码解析:jQuery数据缓存机制详解1
jQuery中有三种添加数据的方法,$().attr(),$().prop(),$().data().但是前面两种是用来在元素上添加属性值的,只适合少量的数据,比如:title,class,name等 ...
- jquery源码解析:addClass,toggleClass,hasClass详解
这一课,我们将继续讲解jQuery对元素属性操作的方法. 首先,我们先看一下这几个方法是如何使用的: $("#div1").addClass("box1 box2&quo ...
随机推荐
- FreeSWITCH 启用多域(多租户)的配置
如果将FreeSWITCH用于云端, 支持大规模并发呼叫, 就要用到 多域/多租户 技术了, FreeSWITCH 本身可以直接支持. 每个域可以单独, 拥有相同的分机号也互相打不通, 各自线路, I ...
- Spark之 使用SparkSql操作mysql和DataFrame的Scala实现
通过读取文件转换成DataFrame数据写入到mysql中 package com.zy.sparksql import java.util.Properties import org.apache. ...
- 【UVA10079 训练指南】收集者的难题【最大流】
题意: Bob和他的朋友从糖果包装里手机贴纸.这些朋友每人手里都有一些(可能有重复的)贴纸,并且只跟别人交换他所没有的贴纸,贴纸总是一对一交换. Bob比这些朋友更聪明,因为他意识到只跟别人交换自己没 ...
- 135. Candy(Array; Greedy)
There are N children standing in a line. Each child is assigned a rating value. You are giving candi ...
- 字符串匹配——C++使用Regex
需要#include < regex > 匹配 regex_match ("subject", std::regex("(sub)(.*)") ...
- Python手机开发调用DLL实现部分ADB功能-乾颐堂
近期学了一点Python,然后正好有一个手机同步工具方面的预研工作要完成. 要实现PC与手机的通信,首先要找到他们的通信协议,还好的是Android有完善的协议:ADB ADB的代码是开源的,而且支持 ...
- g2o 图优化
http://www.cnblogs.com/gaoxiang12/p/5244828.html 扩展里面csparse
- [Email] 收发邮件的协议 : IMAP and SMTP , POP3 and SMTP
支持 IMAP 和 SMTP 的应用 与仅同步收件箱的 POP 不同,IMAP 同步所有电子邮件文件夹. 在电子邮件应用中使用以下设置. 接收 (IMAP) 服务器 服务器地址:imap-mail.o ...
- tomcat启动报错:java.lang.IllegalArgumentException: Document base D:\apache-tomcat-6.0.45\webapps\activiti-explorer does not exist or is not a readable directory
java.lang.IllegalArgumentException: Document base D:\apache-tomcat-6.0.45\webapps\erp does not exist ...
- mysql 字符串操作
-- 字符串的长度 SELECT LENGTH('abc'),LENGTH('我的家'); SELECT CHAR_LENGTH('abc'),CHAR_LENGTH('我的家'); -- 合并字符串 ...