摘要:原本是中用來(lái)當(dāng)作選擇器的,后來(lái)被多帶帶分離出去,成為一個(gè)多帶帶的項(xiàng)目,可以直接導(dǎo)入到項(xiàng)目中使用。。本來(lái)我們使用當(dāng)作選擇器,選定一些或,使用或就可以很快鎖定所在的位置,然后返回給當(dāng)作對(duì)象。的優(yōu)勢(shì)使用的是從右向左的選擇方式,這種方式效率更高。
歡迎來(lái)我的專(zhuān)欄查看系列文章。
Sizzle 原本是 jQuery 中用來(lái)當(dāng)作 DOM 選擇器的,后來(lái)被 John Resig 多帶帶分離出去,成為一個(gè)多帶帶的項(xiàng)目,可以直接導(dǎo)入到項(xiàng)目中使用。jquery/sizzle。
本來(lái)我們使用 jQuery 當(dāng)作選擇器,選定一些 #id 或 .class,使用 document.getElementById 或 document.getElemensByClassName 就可以很快鎖定 DOM 所在的位置,然后返回給 jQuery 當(dāng)作對(duì)象。但有時(shí)候會(huì)碰到一些比較復(fù)雜的選擇 div div.hot>span 這類(lèi)肯定用上面的函數(shù)是不行的,首先考慮到的是 Element.querySelectorAll() 函數(shù),但這個(gè)函數(shù)存在嚴(yán)重的兼容性問(wèn)題MDN querySelectorAll。這個(gè)時(shí)候 sizzle 就派上用場(chǎng)了。
init 函數(shù)介紹中已經(jīng)說(shuō)明白,沒(méi)有介紹 find 函數(shù),其本質(zhì)上就是 Sizzle 函數(shù)在 jQuery 中的表現(xiàn)。這個(gè)函數(shù)在 jQuery 中兩種存在形式,即原型和屬性上分別有一個(gè),先來(lái)看下 jQuery.fn.find:
jQuery.fn.find = function (selector) { var i, ret, len = this.length, self = this; // 這段話(huà)真不知道是個(gè)什么的 if (typeof selector !== "string") { // fn.pushStack 和 jquery.merge 很像,但是返回一個(gè) jquery 對(duì)象,且 // jquery 有個(gè) 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 函數(shù) jQuery.find(selector, self[i], ret); } // uniqueSort 去重函數(shù) return len > 1 ? jQuery.uniqueSort(ret) : ret; }
jQuery.fn.find 的用法一般在 $(".test").find("span"),所以此時(shí)的 this 是指向 $(".test") 的,懂了這一點(diǎn),后面的東西自然而然就好理解了。
然后就是 jQuery.find 函數(shù),本章的重點(diǎn)討論部分。先來(lái)看一個(gè)正則表達(dá)式:
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
你可能會(huì)疑惑,rquickExpr 的名字已經(jīng)出現(xiàn)過(guò)一次了。實(shí)際上 Sizzle 是一個(gè)閉包,這個(gè) rquickExpr 變量是在 Sizzle 閉包內(nèi)的,不會(huì)影響到 jQuery 全局。這個(gè)正則的作用主要是用來(lái)區(qū)分 tag、id 和 class,而且從返回的數(shù)組也有一定的規(guī)律,可以通過(guò)這個(gè)規(guī)律來(lái)判斷 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 函數(shù)其實(shí)是用來(lái)將 context 設(shè)置成 document,考慮到瀏覽器的兼容性 setDocument(context); } context = context || document; // true if (documentIsHTML) { // match 就是那個(gè)有規(guī)律的數(shù)組 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 函數(shù)和 tokenize 函數(shù)后文再談 return select(selector.replace(rtrim, "$1"), context, results, seed); }
整個(gè)分析過(guò)程由于要考慮各種因素,包括效率和瀏覽器兼容性等,所以看起來(lái)非常長(zhǎng),但是邏輯一點(diǎn)都不難:先判斷 selector 是否是非 string,然后正則 rquickExpr 對(duì) selector 進(jìn)行匹配,獲得數(shù)組依次考慮 id、tagName 和 class 情況,這些都很簡(jiǎn)單,都是單一的選擇,一般用瀏覽器自帶的函數(shù) getElement 即可解決。遇到復(fù)雜一點(diǎn)的,比如 div div.show p,先考慮 querySelectorAll 函數(shù)是否支持,然后考慮瀏覽器兼容 IE<8。若不支持,即交給 select 函數(shù)(下章)。
Sizzle 的優(yōu)勢(shì)Sizzle 使用的是從右向左的選擇方式,這種方式效率更高。
瀏覽器在處理 html 的時(shí)候,先生成一個(gè) DOM tree,解析完 css 之后,然后更加 css 和 DOM tess 生成一個(gè) render tree。render tree 用于渲染,不是一一對(duì)應(yīng),如 display:none 的 DOM 就不會(huì)出現(xiàn)在 render tree 中。
如果從左到右的匹配方式,div div.show p,
找到 div 節(jié)點(diǎn),
從 1 的子節(jié)點(diǎn)中找到 div 且 class 為 show 的 DOM,找不到則返回上一步
從 2 的子節(jié)點(diǎn)中找到 p 元素,找不到則返回上一步
如果有一步找不到,向上回溯,直到遍歷所有的 div,效率很低。
如果從右到左的方式,
先匹配到所有的 p 節(jié)點(diǎn),
對(duì) 1 中的結(jié)果注意判斷,若其父節(jié)點(diǎn)順序出現(xiàn) div.show 和 div,則保留,否則丟棄
因?yàn)樽庸?jié)點(diǎn)可以有若干個(gè),而父節(jié)點(diǎn)只有一個(gè),故從右向左的方式效率很高。
衍生的函數(shù) jQuery.fn.pushStackjQuery.fn.pushStack是一個(gè)類(lèi)似于 jQuery.merge 的函數(shù),它接受一個(gè)參數(shù),把該參數(shù)(數(shù)組)合并到一個(gè) jQuery 對(duì)象中并返回,源碼如下:
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
這個(gè)函數(shù)是對(duì) DOM 判斷是否是父子關(guān)系,源碼如下:
jQuery.contains = function (context, elem) { // 考慮到兼容性,設(shè)置 context 的值 if ((context.ownerDocument || context) !== document) { setDocument(context); } return contains(context, elem); } // contains 是內(nèi)部函數(shù),判斷 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 的去重函數(shù),但這個(gè)去重職能處理 DOM 元素?cái)?shù)組,不能處理字符串或數(shù)字?jǐn)?shù)組,來(lái)看看有什么特別的:
jQuery.uniqueSort = function (results) { var elem, duplicates = [], j = 0, i = 0; // hasDuplicate 是一個(gè)判斷是否有相同元素的 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 用于將重復(fù)的元素刪除 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 函數(shù)如下,需要將兩個(gè)函數(shù)放在一起理解才能更明白哦:
var sortOrder = function (a, b) { // 表示有相同的元素,設(shè)置 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; }總結(jié)
可以說(shuō)今天先對(duì) Sizzle 開(kāi)個(gè)頭,任重而道遠(yuǎn)!下面就會(huì)接受 Sizzle 中的 tokens 和 select 函數(shù)。
參考jQuery 2.0.3 源碼分析Sizzle引擎 - 詞法解析
本文在 github 上的源碼地址,歡迎來(lái) star。
歡迎來(lái)我的博客交流。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://specialneedsforspecialkids.com/yun/81551.html
摘要:歡迎來(lái)我的專(zhuān)欄查看系列文章。我們以為例,這是一個(gè)很簡(jiǎn)單的,逗號(hào)將表達(dá)式分成兩部分。這是針對(duì)于存在的情況,對(duì)于不存在的情況,其就是的操作,后面會(huì)談到。參考源碼分析引擎詞法解析選擇器參考手冊(cè)本文在上的源碼地址,歡迎來(lái)。 歡迎來(lái)我的專(zhuān)欄查看系列文章。 在編譯原理中,詞法分析是一個(gè)非常關(guān)鍵的環(huán)節(jié),詞法分析器讀入字節(jié)流,然后根據(jù)關(guān)鍵字、標(biāo)識(shí)符、標(biāo)點(diǎn)、字符串等進(jìn)行劃分,生成單詞。Sizzle 選擇...
摘要:生成終極匹配器主要是返回一個(gè)匿名函數(shù),在這個(gè)函數(shù)中,利用方法生成的匹配器,去驗(yàn)證種子集合,篩選出符合條件的集合。在這個(gè)終極匹配器中,會(huì)將獲取到的種子元素集合與匹配器進(jìn)行比對(duì),篩選出符合條件的元素。 讀Sizzle的源碼,分析的Sizzle版本號(hào)是2.3.3。 Sizzle的Github主頁(yè) 瀏覽器原生支持的元素查詢(xún)方法: 方法名 方法描述 兼容性描述 getElementBy...
摘要:源碼中接受個(gè)參數(shù),空參數(shù),這個(gè)會(huì)直接返回一個(gè)空的對(duì)象,。,這是一個(gè)標(biāo)準(zhǔn)且常用法,表示一個(gè)選擇器,這個(gè)選擇器通常是一個(gè)字符串,或者等,表示選擇范圍,即限定作用,可為,對(duì)象。,會(huì)把普通的對(duì)象或?qū)ο蟀b在對(duì)象中。介紹完入口,就開(kāi)始來(lái)看源碼。 歡迎來(lái)我的專(zhuān)欄查看系列文章。 init 構(gòu)造器 前面一講總體架構(gòu)已經(jīng)介紹了 jQuery 的基本情況,這一章主要來(lái)介紹 jQuery 的入口函數(shù) jQu...
摘要:歡迎來(lái)我的專(zhuān)欄查看系列文章。現(xiàn)在我們?cè)賮?lái)理一理數(shù)組,這個(gè)數(shù)組目前是一個(gè)多重?cái)?shù)組,現(xiàn)在不考慮逗號(hào)的情況,暫定只有一個(gè)分支。源碼源碼之前,來(lái)看幾個(gè)正則表達(dá)式。 歡迎來(lái)我的專(zhuān)欄查看系列文章。 select 函數(shù) 前面已經(jīng)介紹了 tokensize 函數(shù)的功能,已經(jīng)生成了一個(gè) tokens 數(shù)組,而且對(duì)它的組成我們也做了介紹,下面就是介紹對(duì)這個(gè) tokens 數(shù)組如何處理。 showImg(h...
摘要:一種比較合理的方法就是對(duì)應(yīng)每個(gè)可判斷的生成一個(gè)閉包函數(shù),統(tǒng)一進(jìn)行查找。根據(jù)關(guān)系編譯閉包函數(shù),為四組編譯函數(shù)主要借助和。第四步將所有的編譯閉包函數(shù)放到一起,生成函數(shù)。 歡迎來(lái)我的專(zhuān)欄查看系列文章。 compile 講了這么久的 Sizzle,總感覺(jué)差了那么一口氣,對(duì)于一個(gè) selector,我們把它生成 tokens,進(jìn)行優(yōu)化,優(yōu)化的步驟包括去頭和生成 seed 集合。對(duì)于這些種子集合,...
閱讀 1090·2021-09-22 15:19
閱讀 1707·2021-08-23 09:46
閱讀 2231·2021-08-09 13:47
閱讀 1411·2019-08-30 15:55
閱讀 1414·2019-08-30 15:55
閱讀 1978·2019-08-30 15:54
閱讀 2801·2019-08-30 15:53
閱讀 717·2019-08-30 11:03