摘要:歡迎來我的專欄查看系列文章。我們以為例,這是一個很簡單的,逗號將表達式分成兩部分。這是針對于存在的情況,對于不存在的情況,其就是的操作,后面會談到。參考源碼分析引擎詞法解析選擇器參考手冊本文在上的源碼地址,歡迎來。
歡迎來我的專欄查看系列文章。
在編譯原理中,詞法分析是一個非常關鍵的環節,詞法分析器讀入字節流,然后根據關鍵字、標識符、標點、字符串等進行劃分,生成單詞。Sizzle 選擇器的匹配思路和這非常像,在內部叫做 Tokens。
Tokens 詞法分析其實詞法分析是匯編里面提到的詞匯,把它用到這里感覺略有不合適,但 Sizzle 中的 tokensize函數干的就是詞法分析的活。
上一章我們已經講到了 Sizzle 的用法,實際上就是 jQuery.find 函數,只不過還涉及到 jQuery.fn.find。jQuery.find 函數考慮的很周到,對于處理 #id、.class 和 TagName 的情況,都比較簡單,通過一個正則表達式 rquickExpr 將內容給分開,如果瀏覽器支持 querySelectorAll,那更是最好的。
比較難的要數這種類似于 css 選擇器的 selector,div > div.seq h2 ~ p , #id p,如果使用從左向右的查找規則,效率很低,而從右向左,可以提高效率。
本章就來介紹 tokensize 函數,看看它是如何將復雜的 selector 處理成 tokens 的。
我們以 div > div.seq h2 ~ p , #id p 為例,這是一個很簡單的 css,逗號 , 將表達式分成兩部分。css 中有一些基本的符號,這里有必要強調一下,比如 , space > + ~:
div,p , 表示并列關系,所有 div 元素和 p 元素;
div p 空格表示后代元素,div 元素內所有的 p 元素;
div>p > 子元素且相差只能是一代,父元素為 div 的所有 p 元素;
div+p + 表示緊鄰的兄弟元素,前一個兄弟節點為 div 的所有 p 元素;
div~p ~ 表示兄弟元素,所有前面有兄弟元素 div 的所有 p 元素。
除此之外,還有一些 a、input 比較特殊的:
a[target=_blank] 選擇所有 target 為 _blank 的所有 a 元素;
a[title=search] 選擇所有 title 為 search 的所有 a 元素;
input[type=text] 選擇 type 為 text 的所有 input 元素;
p:nth-child(2) 選擇其為父元素第二個元素的所有 p 元素;
Sizzle 都是支持這些語法的,如果我們把這一步叫做詞法分析,那么詞法分析的結果是一個什么東西呢?
div > div.seq h2 ~ p , #id p 經過 tokensize(selector) 會返回一個數組,該數組在函數中稱為 groups。因為這個例子有一個逗號,故該數組有兩個元素,分別是 tokens[0] 和 tokens[1],代表選擇器逗號前后的兩部分。tokens 也是數組,它的每一個元素都是一個 token 對象。
一個 token 對象結構如下所示:
token: { value: matched, // 匹配到的字符串 type: type, //token 類型 matches: match //去除 value 的正則結果數組 }
Sizzle 中 type 的種類有下面幾種:ID、CLASS、TAG、ATTR、PSEUDO、CHILD、bool、needsContext,這幾種有幾種我也不知道啥意思,child 表示 nth-child、even、odd 這種子選擇器。這是針對于 matches 存在的情況,對于 matches 不存在的情況,其 type 就是 value 的 trim() 操作,后面會談到。
tokensize 函數對 selector 的處理,連空格都不放過,因為空格也屬于 type 的一種,而且還很重要,div > div.seq h2 ~ p 的處理結果:
tokens: [ [value:"div", type:"TAG", matches:Array[1]], [value:" > ", type:">"], [value:"div", type:"TAG", matches:Array[1]], [value:".seq", type:"CLASS", matches:Array[1]], [value:" ", type:" "], [value:"h2", type:"TAG", matches:Array[1]], [value:" ~ ", type:"~"], [value:"p", type:"TAG", matches:Array[1]], ]
這個數組會交給 Sizzle 的下一個流程來處理,今天暫不討論。
tokensize 源碼照舊,先來看一下幾個正則表達式。
var rcomma = /^[x20 f]*,[x20 f]*/; rcomma.exec("div > div.seq h2 ~ p");//null rcomma.exec(" ,#id p");//[" ,"]
rcomma 這個正則,主要是用來區分 selector 是否到下一個規則,如果到下一個規則,就把之前處理好的 push 到 groups 中。這個正則中 [x20 f] 是用來匹配類似于 whitespace 的,主體就一個逗號。
var rcombinators = /^[x20 f]*([>+~]|[x20 f])[x20 f]*/; rcombinators.exec(" > div.seq h2 ~ p"); //[" > ", ">"] rcombinators.exec(" ~ p"); //[" ~ ", "~"] rcombinators.exec(" h2 ~ p"); //[" ", " "]
是不是看來 rcombinators 這個正則表達式,上面 tokens 那個數組的內容就完全可以看得懂了。
其實,如果看 jQuery 的源碼,rcomma 和 rcombinators 并不是這樣來定義的,而是用下面的方式來定義:
var whitespace = "[x20 f]"; var rcomma = new RegExp( "^" + whitespace + "*," + whitespace + "*" ), rcombinators = new RegExp( "^" + whitespace + "*([>+~]|" + whitespace + ")" + whitespace + "*" ), rtrim = new RegExp( "^" + whitespace + "+|((?:^|[^])(?:.)*)" + whitespace + "+$", "g" ),
有的時候必須得要佩服 jQuery 中的做法,該簡則簡,該省則省,每一處代碼都是極完美的。
還有兩個對象,Expr 和 matchExpr,Expr 是一個非常關鍵的對象,它涵蓋了幾乎所有的可能的參數,比較重要的參數比如有:
Expr.filter = { "TAG": function(){...}, "CLASS": function(){...}, "ATTR": function(){...}, "CHILD": function(){...}, "ID": function(){...}, "PSEUDO": function(){...} } Expr.preFilter = { "ATTR": function(){...}, "CHILD": function(){...}, "PSEUDO": function(){...} }
這個 filter 和 preFilter 是處理 type=TAG 的關鍵步驟,包括一些類似于 input[type=text] 也是這幾個函數處理,也比較復雜,我本人是看迷糊了。還有 matchExpr 正則表達式:
var identifier = "(?:.|[w-]|[^