首页 > 语言 > JavaScript > 正文

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

2024-05-06 15:11:41
字体:
来源:转载
供稿:网友

前言

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);}            
发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表

图片精选