jQuery源码分析之sizzle选择器详解
发布时间 - 2026-01-10 23:00:50 点击率:次前言

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 具体是哪一种。
jQuery.find = Sizzle;
function Sizzle(selector, context, results, seed) {
var m, i, elem, nid, match, groups, newSelector, newContext = context && context.ownerDocument,
// nodeType defaults to 9, since context defaults to document
nodeType = context ? context.nodeType : 9;
results = results || [];
// Return early from calls with invalid selector or context
if (typeof selector !== "string" || !selector || nodeType !== 1 && nodeType !== 9 && nodeType !== 11) {
return results;
}
// Try to shortcut find operations (as opposed to filters) in HTML documents
if (!seed) {
if ((context ? context.ownerDocument || context : preferredDoc) !== document) {
// setDocument 函数其实是用来将 context 设置成 document,考虑到浏览器的兼容性
setDocument(context);
}
context = context || document;
// true
if (documentIsHTML) {
// match 就是那个有规律的数组
if (nodeType !== 11 && (match = rquickExpr.exec(selector))) {
// selector 是 id 的情况
if ((m = match[1])) {
// Document context
if (nodeType === 9) {
if ((elem = context.getElementById(m))) {
if (elem.id === m) {
results.push(elem);
return results;
}
} else {
return results;
}
// 非 document 的情况
} else {
if (newContext && (elem = newContext.getElementById(m)) && contains(context, elem) && elem.id === m) {
results.push(elem);
return results;
}
}
// selector 是 tagName 情况
} else if (match[2]) {
// 这里的 push:var push = arr.push
push.apply(results, context.getElementsByTagName(selector));
return results;
// selector 是 class 情况
} else if ((m = match[3]) && support.getElementsByClassName && context.getElementsByClassName) {
push.apply(results, context.getElementsByClassName(m));
return results;
}
}
// 如果浏览器支持 querySelectorAll
if (support.qsa && !compilerCache[selector + " "] && (!rbuggyQSA || !rbuggyQSA.test(selector))) {
if (nodeType !== 1) {
newContext = context;
newSelector = selector;
// qSA looks outside Element context, which is not what we want
// Support: IE <=8,还是要考虑兼容性
} else if (context.nodeName.toLowerCase() !== "object") {
// Capture the context ID, setting it first if necessary
if ((nid = context.getAttribute("id"))) {
nid = nid.replace(rcssescape, fcssescape);
} else {
context.setAttribute("id", (nid = expando));
}
// Sizzle 词法分析的部分
groups = tokenize(selector);
i = groups.length;
while (i--) {
groups[i] = "#" + nid + " " + toSelector(groups[i]);
}
newSelector = groups.join(",");
// Expand context for sibling selectors
newContext = rsibling.test(selector) && testContext(context.parentNode) || context;
}
if (newSelector) {
try {
push.apply(results, newContext.querySelectorAll(newSelector));
return results;
} catch(qsaError) {} finally {
if (nid === expando) {
context.removeAttribute("id");
}
}
}
}
}
}
// All others,select 函数和 tokenize 函数后文再谈
return select(selector.replace(rtrim, "$1"), context, results, seed);
}
整个分析过程由于要考虑各种因素,包括效率和浏览器兼容性等,所以看起来非常长,但是逻辑一点都不难:先判断 selector 是否是非 string,然后正则 rquickExpr 对 selector 进行匹配,获得数组依次考虑 id、tagName 和 class 情况,这些都很简单,都是单一的选择,一般用浏览器自带的函数 getElement 即可解决。遇到复杂一点的,比如 div div.show p,先考虑 querySelectorAll 函数是否支持,然后考虑浏览器兼容 IE<8。若不支持,即交给 select 函数(下章)。
Sizzle 的优势
Sizzle 使用的是从右向左的选择方式,这种方式效率更高。
浏览器在处理 html 的时候,先生成一个 DOM tree,解析完 css 之后,然后更加 css 和 DOM tess 生成一个 render tree。render tree 用于渲染,不是一一对应,如 display:none 的 DOM 就不会出现在 render tree 中。
如果从左到右的匹配方式,div div.show p,
- 找到 div 节点,
- 从 1 的子节点中找到 div 且 class 为 show 的 DOM,找不到则返回上一步
- 从 2 的子节点中找到 p 元素,找不到则返回上一步
如果有一步找不到,向上回溯,直到遍历所有的 div,效率很低。
如果从右到左的方式,
- 先匹配到所有的 p 节点,
- 对 1 中的结果注意判断,若其父节点顺序出现
div.show和 div,则保留,否则丢弃
因为子节点可以有若干个,而父节点只有一个,故从右向左的方式效率很高。
衍生的函数
jQuery.fn.pushStack
jQuery.fn.pushStack是一个类似于 jQuery.merge 的函数,它接受一个参数,把该参数(数组)合并到一个 jQuery 对象中并返回,源码如下:
jQuery.fn.pushStack = function (elems) {
// Build a new jQuery matched element set
var ret = jQuery.merge(this.constructor(), elems);
// Add the old object onto the stack (as a reference)
ret.prevObject = this;
// Return the newly-formed element set
return ret;
}
jQuery.contains
这个函数是对 DOM 判断是否是父子关系,源码如下:
jQuery.contains = function (context, elem) {
// 考虑到兼容性,设置 context 的值
if ((context.ownerDocument || context) !== document) {
setDocument(context);
}
return contains(context, elem);
}
// contains 是内部函数,判断 DOM_a 是否是 DOM_b 的
var contains = function (a, b) {
var adown = a.nodeType === 9 ? a.documentElement : a,
bup = b && b.parentNode;
return a === bup || !!(bup && bup.nodeType === 1 && (
adown.contains ? adown.contains(bup) : a.compareDocumentPosition && a.compareDocumentPosition(bup) & 16));
}
jQuery.uniqueSort
jQuery 的去重函数,但这个去重职能处理 DOM 元素数组,不能处理字符串或数字数组,来看看有什么特别的:
jQuery.uniqueSort = function (results) {
var elem, duplicates = [],
j = 0,
i = 0;
// hasDuplicate 是一个判断是否有相同元素的 flag,全局
hasDuplicate = !support.detectDuplicates;
sortInput = !support.sortStable && results.slice(0);
results.sort(sortOrder);
if (hasDuplicate) {
while ((elem = results[i++])) {
if (elem === results[i]) {
j = duplicates.push(i);
}
}
while (j--) {
// splice 用于将重复的元素删除
results.splice(duplicates[j], 1);
}
}
// Clear input after sorting to release objects
// See https://github.com/jquery/sizzle/pull/225
sortInput = null;
return results;
}
sortOrder 函数如下,需要将两个函数放在一起理解才能更明白哦:
var sortOrder = function (a, b) {
// 表示有相同的元素,设置 flag 为 true
if (a === b) {
hasDuplicate = true;
return 0;
}
// Sort on method existence if only one input has compareDocumentPosition
var compare = !a.compareDocumentPosition - !b.compareDocumentPosition;
if (compare) {
return compare;
}
// Calculate position if both inputs belong to the same document
compare = (a.ownerDocument || a) === (b.ownerDocument || b) ? a.compareDocumentPosition(b) :
// Otherwise we know they are disconnected
1;
// Disconnected nodes
if (compare & 1 || (!support.sortDetached && b.compareDocumentPosition(a) === compare)) {
// Choose the first element that is related to our preferred document
if (a === document || a.ownerDocument === preferredDoc && contains(preferredDoc, a)) {
return -1;
}
if (b === document || b.ownerDocument === preferredDoc && contains(preferredDoc, b)) {
return 1;
}
// Maintain original order
return sortInput ? (indexOf(sortInput, a) - indexOf(sortInput, b)) : 0;
}
return compare & 4 ? -1 : 1;
}
总结
可以说今天先对 Sizzle 开个头,任重而道远!下面就会接受 Sizzle 中的 tokens 和 select 函数。感兴趣的朋友们可以继续关注,希望本文的内容对大家能有一定的帮助。
# jquery
# sizzle
# sizzle选择器
# 选择器
# jquery 选择器引擎sizzle浅析
# jQuery源码分析-04 选择器-Sizzle-工作原理分析
# jQuery选择器源码解读(一):Sizzle方法
# jQuery选择器源码解读(六):Sizzle选择器匹配逻辑分析
# 是一个
# 找不到
# 考虑到
# 有一定
# 的是
# 都是
# 判断是否
# 上一步
# 有什么
# 就会
# 是个
# 是在
# 在这里
# 有个
# 中找到
# 任重而道远
# 是用来
# 可以说
# 出现在
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
JavaScript如何实现继承_有哪些常用方法
香港网站服务器数量如何影响SEO优化效果?
html如何与html链接_实现多个HTML页面互相链接【互相】
在centOS 7安装mysql 5.7的详细教程
Laravel队列任务超时怎么办_Laravel Queue Timeout设置详解
Android Socket接口实现即时通讯实例代码
如何在Windows 2008云服务器安全搭建网站?
如何基于PHP生成高效IDC网络公司建站源码?
海南网站制作公司有哪些,海口网是哪家的?
如何在建站宝盒中设置产品搜索功能?
Laravel怎么使用Markdown渲染文档_Laravel将Markdown内容转HTML页面展示【实战】
安克发布新款氮化镓充电宝:体积缩小 30%,支持 200W 输出
C++时间戳转换成日期时间的步骤和示例代码
黑客如何利用漏洞与弱口令入侵网站服务器?
php打包exe后无法访问网络共享_共享权限设置方法【教程】
香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化
如何制作新型网站程序文件,新型止水鱼鳞网要拆除吗?
如何在橙子建站上传落地页?操作指南详解
Laravel如何使用Blade模板引擎?(完整语法和示例)
Laravel如何实现用户注册和登录?(Auth脚手架指南)
zabbix利用python脚本发送报警邮件的方法
Laravel的HTTP客户端怎么用_Laravel HTTP Client发起API请求教程
ChatGPT 4.0官网入口地址 ChatGPT在线体验官网
Laravel怎么实现API接口鉴权_Laravel Sanctum令牌生成与请求验证【教程】
如何用PHP快速搭建高效网站?分步指南
网站制作壁纸教程视频,电脑壁纸网站?
广州网站制作公司哪家好一点,广州欧莱雅百库网络科技有限公司官网?
详解Huffman编码算法之Java实现
如何做网站制作流程,*游戏网站怎么搭建?
Laravel项目怎么部署到Linux_Laravel Nginx配置详解
Laravel控制器是什么_Laravel MVC架构中Controller的作用与实践
Laravel请求验证怎么写_Laravel Validator自定义表单验证规则教程
如何在阿里云完成域名注册与建站?
Laravel Asset编译怎么配置_Laravel Vite前端构建工具使用
作用域操作符会触发自动加载吗_php类自动加载机制与::调用【教程】
PHP正则匹配日期和时间(时间戳转换)的实例代码
html文件怎么打开证书错误_https协议的html打开提示不安全【指南】
Laravel如何实现图片防盗链功能_Laravel中间件验证Referer来源请求【方案】
Windows10怎样连接蓝牙设备_Windows10蓝牙连接步骤【教程】
HTML5段落标签p和br怎么选_文本排版常用标签对比【解答】
谷歌Google入口永久地址_Google搜索引擎官网首页永久入口
小视频制作网站有哪些,有什么看国内小视频的网站,求推荐?
学生网站制作软件,一个12岁的学生写小说,应该去什么样的网站?
LinuxShell函数封装方法_脚本复用设计思路【教程】
python中快速进行多个字符替换的方法小结
Laravel API路由如何设计_Laravel构建RESTful API的路由最佳实践
猎豹浏览器开发者工具怎么打开 猎豹浏览器F12调试工具使用【前端必备】
企业在线网站设计制作流程,想建设一个属于自己的企业网站,该如何去做?
Python文件异常处理策略_健壮性说明【指导】
HTML5打空格有哪些误区_新手常犯的空格使用错误【技巧】

