写在开头:
昨天开始,我决定要认真的看看jQuery的源码,选择1.7.2,源于公司用的这个版本。由于源码比较长,这将会是一个比较持久的过程,我将要利用业余时间,和偶尔上班不算忙的时间来进行。其实原本是打算对着源码抄一遍,将对其的理解写成注释,这也算是在强行堆代码量了吧(我想我这是有多懒,必须要反省)。不过鉴于自己平时比较懒惰的可耻行径,和太多的东西都写在一起有点庞大,我想我还是有必要写成一个专栏,来记录这个过程。其实最根本的原因是:源码里都是有注释的,而且注释写得那么详尽,翻译过来就是了,但是是否真正的理解了呢?还有它涉及到的其他的知识点,都是有必要去研究一下的。
正文开始:
首先,抬头的注释里有一句 Includes Sizzle.js ,关于Sizzle.js,后面再来研究。
整个jQuery的代码是写在一个(function(){})(window);的一个闭包函数里,用一个用匿名函数并加()进行运行,统一命名空间,防止变量的污染。匿名函数定义了两个参数window, undefined,然后在执行时将window和window.undefined传给函数。将window作为参数传入,在函数里就可以直接调用window,而不必去找最外层的对象,这样在对window进行操作或者调用window的一些方法、属性时效率要高一些。Javascript 中的 undefined 并不是作为关键字,使用参数undefined 是为了防外面定义undefined变量而受污染(调用时第二个参数未写,即是传入了最原始的window.undefied)。
然后再是定义一些变量存储常用的一些属性和方法(同样是为了便于不必每次都去查找)。
在jQuery的构造函数(这样讲比较明了)返回的是一个jQuery.fn.init的对象。也就是说,new jQuery()和jQuery()返回的都是jQuery.fn.init的对象,所以我们在使用jQuery的时候,通常是直接$()的方式创建jQuery对象。然后通过jQuery.fn.init.prototype=jQuery.fn=jQuery.prototype来将jQuery的原型和jQuery.fn.init的原型联系在一起,这样的写法看着会有点绕,但是其实也就是为了让返回的jQuery.fn.init的对象可以调用jQuery.prototype里定义的那些方法和属性。
在init里主要就是针对不同的选择器返回对应的jQuery.fn.init对象。
/*!
* ...省略...
* Includes Sizzle.js
* http://sizzlejs.com/
* ...省略...
*/
(function (window, undefined) {
//用变量存储window的属性
var document = window.document,
navigator = window.navigator,
location = window.location;
//定义jQuery函数,并立即执行(//TODO)
var jQuery = (function () { //定义一个局部变量,它对应jQuery(最终被return)
var jQuery = function (selector, context) { //返回一个jQuery.fn.init对象(返回的是一个new之后的对象,
//这样就可以避免创建jQuery对象的时候写成 new jQuery()或者new $()...)
return new jQuery.fn.init(selector, context, rootjQuery);
}, //存储window.jQuery,以防万一被覆盖
_jQuery = window.jQuery,
_$ = window.$, //指向root jquery的引用
rootjQuery, //...省略一系列的正则...
//供于jQuery.camelCase回调,转换成驼峰式
fcamelCase = function (all, letter) {
return (letter + "").toUpperCase();
}, //存储用户代理以便jQuery.browser使用(为了判断浏览器的类型及版本)
userAgent = navigator.userAgent, //用来匹配引擎和浏览器版本
browserMatch, //ready事件(dom加载完成之后)处理函数队列
readyList, // The ready event handler
DOMContentLoaded, // Save a reference to some core methods
toString = Object.prototype.toString,
hasOwn = Object.prototype.hasOwnProperty,
push = Array.prototype.push,
slice = Array.prototype.slice,//slice(start,end) 方法可从已有的数组中返回选定的元素。
trim = String.prototype.trim,
indexOf = Array.prototype.indexOf, // [[Class]] -> type pairs //TODO-暂时我也不知干嘛的
class2type = {}; //定义jQuery.prototype,并赋给jQuery.fn(用jQuery.fn写jQuery扩展的原因所在)
jQuery.fn = jQuery.prototype = {
constructor: jQuery,
init: function (selector, context, rootjQuery) {
var match, elem, ret, doc; // Handle $(""), $(null), or $(undefined) ->返回本身
if (!selector) {
return this;
} // Handle $(DOMElement) -> 如果是element ,就直接封装成jquery对象
if (selector.nodeType) {
this.context = this[0] = selector;
this.length = 1;
return this;
} //如果是body ->这里document.body作为条件,应该是为了兼容问题
//可是今天我在主流浏览器里测试,都能够同时找到document.body和document.documentElement
if (selector === "body" && !context && document.body) {
this.context = document;
this[0] = document.body;
this.selector = selector;
return this;
} if (typeof selector === "string") {
//HTML string
if (selector.charAt(0) === "<" && selector.charAt(selector.length - 1) === ">" && selector.length >= 3) { //跳过检查
match = [null, selector, null];
} else {
//RegExpObject.exec(string)返回一个数组,其中存放匹配的结果。如果未找到匹配,则返回值为 null。
match = quickExpr.exec(selector);
} //验证match, match[1](这里存放是html)为非的时候context也必须为非(这种情况是#id)
if (match && (match[1] || !context)) { // HANDLE: $(html) -> $(array)
if (match[1]) {
//这里的context其实可以理解成selector的parentNode或者parent()
//context ->DOM对象
context = context instanceof jQuery ? context[0] : context;
//如果制定了context,就返回context.ownerDocument(这里是context当前所属的document) || context,否则返回document
doc = (context ? context.ownerDocument || context : document); //匹配成独立的标签(不含有属性之类,比如<a></a>)
ret = rsingleTag.exec(selector); if (ret) {
//方法jQuery.isPlainObject( object )用于判断传入的参数是否是“纯粹”的对象,即是否是用对象直接量{}或new Object()创建的对象
if (jQuery.isplainObject(context)) {
selector = [document.createElement(ret[1])];
jQuery.fn.attr.call(selector, context, true);
} else {
selector = [doc.createElement(ret[1])];
}
} else {
//缓存selector的html。
ret = jQuery.buildFragment([match[1]], [doc]);
//如果是缓存了的,就clone fragment(文档碎片节点在添加到文档树之后便不能再对其进行操作),否则就直接取fragment 的childNodes
selector = (ret.cacheable ? jQuery.clone(ret.fragment) : ret.fragment).childNodes;
}
//将selector合并到this,返回
return jQuery.merge(this, selector); // HANDLE: $("#id")
} else {
elem = document.getElementById(match[2]); if (elem && elem.parentNode) {
//处理 IE and Opera 混淆ID与NAME的bug
if (elem.id !== match[2]) {
//调用Sizzle的方法 -- TODO,关于Sizzle.js,有待研究!
return rootjQuery.find(selector);
} this.length = 1;
this[0] = elem;
} this.context = document;
this.selector = selector;
return this;
} // HANDLE: $(expr, $(...))
} else if (!context || context.jquery) {
return (context || rootjQuery).find(selector); // HANDLE: $(expr, context)
} else {
return this.constructor(context).find(selector);
} // HANDLE: $(function)
} else if (jQuery.isFunction(selector)) {
return rootjQuery.ready(selector);
} //selector本身就是一个jQuery对象的情况
if (selector.selector !== undefined) {
this.selector = selector.selector;
this.context = selector.context;
} //合并属性(与jQuery.merge不同的是,这里的selector可能不是数组)
return jQuery.makeArray(selector, this);
}, // Start with an empty selector
selector: "", // The current version of jQuery being used
jquery: "1.7.2", // The default length of a jQuery object is 0
length: 0,
//TODO--未完待续 }; })();
})(window);
关于DocumentFragment:
文档碎片是一种"轻量级"文档对象,可以包含和控制节点(同其他文档对象一样),但不会像完整的文档那样占用额外资源,在添加大量的节点的情况下,可以将文档碎片当做一个缓存使用,把节点先存放到文档碎片中,再把DocumentFragment节点插入文档树(当把文档碎片插入文档树时,插入的是它的子孙节点),在要添加的节点量很大的情况下效率会要高很多。
参考资料:http://www.cnblogs.com/aaronjs/p/3510768.html#_h1_2
源码下载: http://jquery.com/