jQuery源码分析之sizzle选择器详解

2020-05-23 06:19:08易采站长站整理

前言

Sizzle 原本是 jQuery 中用来当作 DOM 选择器的,后来被 John Resig 单独分离出去,成为一个单独的项目,可以直接导入到项目中使用。

点击这里下:jquery/sizzle。

本来我们使用 jQuery 当作选择器,选定一些 #id 或 .class,使用

document.getElementById 
document.getElemensByClassName 
就可以很快锁定 DOM 所在的位置,然后返回给 jQuery 当作对象。但有时候会碰到一些比较复杂的选择
 div div.hot>span 
这类肯定用上面的函数是不行的,首先考虑到的是
 Element.querySelectorAll() 
函数,但这个函数存在严重的兼容性问题MDN querySelectorAll。这个时候 sizzle 就派上用场了。

init 函数介绍中已经说明白,没有介绍 find 函数,其本质上就是 Sizzle 函数在 jQuery 中的表现。

这个函数在 jQuery 中两种存在形式,即原型和属性上分别有一个,先来看下

jQuery.fn.find
:


jQuery.fn.find = function (selector) {
var i, ret, len = this.length,
self = this;
// 这段话真不知道是个什么的
if (typeof selector !== "string") {
// fn.pushStack 和 jquery.merge 很像,但是返回一个 jquery 对象,且
// jquery 有个 prevObject 属性指向自己
return this.pushStack(jQuery(selector).filter(function () {
for (i = 0; i < len; i++) {
// jQuery.contains(a, b) 判断 a 是否是 b 的父代
if (jQuery.contains(self[i], this)) {
return true;
}
}
}));
}

ret = this.pushStack([]);

for (i = 0; i < len; i++) {
// 在这里引用到 jQuery.find 函数
jQuery.find(selector, self[i], ret);
}
// uniqueSort 去重函数
return len > 1 ? jQuery.uniqueSort(ret) : ret;
}

jQuery.fn.find 
的用法一般在
$('.test').find("span") 
,所以此时的 this 是指向
 $(‘.test') 
的,懂了这一点,后面的东西自然而然就好理解了。

然后就是

jQuery.find 
函数,本章的重点讨论部分。

先来看一个正则表达式:


var rquickExpr = /^(?:#([w-]+)|(w+)|.([w-]+))$/;
rquickExpr.exec('#id') //["#id", "id", undefined, undefined]rquickExpr.exec('div') //["div", undefined, "div", undefined]rquickExpr.exec('.test') //[".test", undefined, undefined, "test"]rquickExpr.exec('div p')// null

你可能会疑惑,rquickExpr 的名字已经出现过一次了。实际上 Sizzle 是一个闭包,这个 rquickExpr 变量是在 Sizzle 闭包内的,不会影响到 jQuery 全局。这个正则的作用主要是用来区分 tag、id 和 class,而且从返回的数组也有一定的规律,可以通过这个规律来判断 selector 具体是哪一种。