亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 語言 > JavaScript > 正文

jQuery選擇器源碼解讀(一):Sizzle方法

2024-05-06 16:17:53
字體:
來源:轉載
供稿:網友

這篇文章主要介紹了jQuery選擇器源碼解讀(一):Sizzle方法,本文用詳細的注釋解讀了Sizzle方法的實現源碼,需要的朋友可以參考下

對jQuery的Sizzle各方法做了深入分析(同時也參考了一些網上資料)后,將結果分享給大家。我將采用連載的方式,對Sizzle使用的一些方法詳細解釋一下,每篇文章介紹一個方法。

若需要轉載,請寫明出處,多謝。
 

  1. /* 
  2. * Sizzle方法是Sizzle選擇器包的主要入口,jQuery的find方法就是調用該方法獲取匹配的節點 
  3. * 該方法主要完成下列任務: 
  4. * 1、對于單一選擇器,且是ID、Tag、Class三種類型之一,則直接獲取并返回結果 
  5. * 2、對于支持querySelectorAll方法的瀏覽器,通過執行querySelectorAll方法獲取并返回匹配的DOM元素 
  6. * 3、除上之外則調用select方法獲取并返回匹配的DOM元素 
  7.  
  8.  
  9. * @param selector 選擇器字符串 
  10. * @param context 執行匹配的最初的上下文(即DOM元素集合)。若context沒有賦值,則取document。 
  11. * @param results 已匹配出的部分最終結果。若results沒有賦值,則賦予空數組。 
  12. * @param seed 初始集合 
  13. */ 
  14. function Sizzle(selector, context, results, seed) { 
  15. var match, elem, m, nodeType, 
  16. // QSA vars 
  17. i, groups, old, nid, newContext, newSelector; 
  18.  
  19. /* 
  20. * preferredDoc = window.document 
  21.  
  22. * setDocument方法完成一些初始化工作 
  23. */ 
  24. if ((context ? context.ownerDocument || context : preferredDoc) !== document) { 
  25. setDocument(context); 
  26.  
  27. context = context || document; 
  28. results = results || []; 
  29.  
  30. /* 
  31. * 若selector不是有效地字符串類型數據,則直接返回results 
  32. */ 
  33. if (!selector || typeof selector !== "string") { 
  34. return results; 
  35.  
  36. /* 
  37. * 若context既不是document(nodeType=9),也不是element(nodeType=1),那么就返回空集合 
  38. */ 
  39. if ((nodeType = context.nodeType) !== 1 && nodeType !== 9) { 
  40. return []; 
  41.  
  42. // 若當前過濾的是HTML文檔,且沒有設定seed,則執行if內的語句體 
  43. if (documentIsHTML && !seed) { 
  44.  
  45. /*  
  46. * 若選擇器是單一選擇器,且是ID、Tag、Class三種類型之一,則直接獲取并返回結果 
  47.  
  48. * rquickExpr = /^(?:#([/w-]+)|(/w+)|/.([/w-]+))$/ 
  49. * 上述正則表達式括號內三段依次分別用來判斷是否是ID、TAG、CLASS類型的單一選擇器 
  50. * 上述正則表達式在最外層圓括號內有三個子表達式(即三個圓括號括起來的部分), 
  51. * 分別代表ID、Tag、Class選擇器的值,在下面代碼中,分別體現在match[1]、match[2]、match[3] 
  52. */ 
  53. if ((match = rquickExpr.exec(selector))) { 
  54. // Speed-up: Sizzle("#ID") 
  55. // 處理ID類型選擇器,如:#ID 
  56. if ((m = match[1])) { 
  57. // 若當前上下文是一個document,則執行if內語句體 
  58. if (nodeType === 9) { 
  59. elem = context.getElementById(m); 
  60. // Check parentNode to catch when Blackberry 4.6 
  61. // returns 
  62. // nodes that are no longer in the document #6963 
  63. if (elem && elem.parentNode) { 
  64. // Handle the case where IE, Opera, and Webkit 
  65. // return items 
  66. // by name instead of ID 
  67. /* 
  68. * 一些老版本的瀏覽器會把name當作ID來處理, 
  69. * 返回不正確的結果,所以需要再一次對比返回節點的ID屬性 
  70. */ 
  71. if (elem.id === m) { 
  72. results.push(elem); 
  73. return results; 
  74. else { 
  75. return results; 
  76. else { 
  77. // Context is not a document 
  78. /* 
  79. * contains(context, elem)用來確認獲取的elem是否是當前context對象的子對象 
  80. */ 
  81. if (context.ownerDocument 
  82. && (elem = context.ownerDocument.getElementById(m)) 
  83. && contains(context, elem) && elem.id === m) { 
  84. results.push(elem); 
  85. return results; 
  86.  
  87. // Speed-up: Sizzle("TAG") 
  88. // 處理Tag類型選擇器,如:SPAN 
  89. else if (match[2]) { 
  90. push.apply(results, context.getElementsByTagName(selector)); 
  91. return results; 
  92.  
  93. // Speed-up: Sizzle(".CLASS") 
  94. /* 
  95. * 處理class類型選擇器,如:.class 
  96. * 下面條件判斷分別是: 
  97. * m = match[3]:有效的class類型選擇器 
  98. * support.getElementsByClassName 該選擇器的div支持getElementsByClassName 
  99. * context.getElementsByClassName 當前上下文節點有getElementsByClassName方法 
  100.  
  101. */ 
  102.  
  103. else if ((m = match[3]) && support.getElementsByClassName 
  104. && context.getElementsByClassName) { 
  105. push.apply(results, context.getElementsByClassName(m)); 
  106. return results; 
  107.  
  108. // QSA path 
  109. /* 
  110. * 若瀏覽器支持querySelectorAll方法且選擇器符合querySelectorAll調用標準,則執行if內語句體 
  111. * 在這里的檢查僅僅是簡單匹配 
  112. * 第一次調用Sizzle時,rbuggyQSA為空 
  113.  
  114. * if語句體內對當前context對象的id的賦值與恢復,是用來修正querySelectorAll的一個BUG 
  115. * 該BUG會在某些情況下把當前節點(context)也作為結果返回回來。 
  116. * 具體方法是,在現有的選擇器前加上一個屬性選擇器:[id=XXX], 
  117. * XXX 為context的id,若context本身沒有設置id,則給個默認值expando。 
  118. */ 
  119.  
  120. if (support.qsa && (!rbuggyQSA || !rbuggyQSA.test(selector))) { 
  121. nid = old = expando; 
  122. newContext = context; 
  123. // 若context是document,則newSelector取自selector,否則為false 
  124. newSelector = nodeType === 9 && selector; 
  125.  
  126. // qSA works strangely on Element-rooted queries 
  127. // We can work around this by specifying an extra ID on the 
  128. // root 
  129. // and working up from there (Thanks to Andrew Dupont for 
  130. // the technique) 
  131. // IE 8 doesn't work on object elements 
  132. if (nodeType === 1 && context.nodeName.toLowerCase() !== "object") { 
  133. groups = tokenize(selector); 
  134.  
  135. if ((old = context.getAttribute("id"))) { 
  136. /* 
  137. * rescape = /'|///g, 
  138. * 這里將old中的單引號、豎杠、反斜杠前加一個反斜杠 
  139. * old.replace(rescape, "//#FormatTableID_0#amp;")代碼中的#FormatTableID_0#amp;代表匹配項 
  140. */ 
  141. nid = old.replace(rescape, "//#FormatTableID_0#amp;"); 
  142. else { 
  143. context.setAttribute("id", nid); 
  144. nid = "[id='" + nid + "'] "
  145.  
  146. // 重新組合新的選擇器 
  147. i = groups.length; 
  148. while (i--) { 
  149. groups[i] = nid + toSelector(groups[i]); 
  150. /* 
  151. * rsibling = new RegExp(whitespace + "*[+~]") 
  152. * rsibling用于判定選擇器是否存在兄弟關系符 
  153. * 若包含+~符號,則取context的父節點作為當前節點 
  154. */ 
  155. newContext = rsibling.test(selector) && context.parentNode 
  156. || context; 
  157. newSelector = groups.join(","); 
  158.  
  159. if (newSelector) { 
  160. /* 
  161. * 這里之所以需要用try...catch, 
  162. * 是因為jquery所支持的一些選擇器是querySelectorAll所不支持的, 
  163. * 當使用這些選擇器時,querySelectorAll會報非法選擇器, 
  164. * 故需要jquery自身去實現。 
  165. */ 
  166. try { 
  167. // 將querySelectorAll獲取的結果并入results,而后返回resulsts 
  168. push.apply(results, newContext 
  169. .querySelectorAll(newSelector)); 
  170. return results; 
  171. catch (qsaError) { 
  172. finally { 
  173. if (!old) { 
  174. context.removeAttribute("id"); 
  175.  
  176. // All others 
  177. // 除上述快捷方式和調用querySelectorAll方式直接獲取結果外,其余都需調用select來獲取結果 
  178. /* 
  179. * rtrim = new RegExp("^" + whitespace + "+|((?:^|[^////])(?:////.)*)" 
  180. * + whitespace + "+$", "g"), 
  181. * whitespace = "[//x20//t//r//n//f]"; 
  182. * 上述rtrim正則表達式的作用是去掉selector兩邊的空白,空白字符由whitespace變量定義 
  183. * rtrim的效果與new RegExp("^" + whitespace + "+|" + whitespace + "+$", "g")相似 
  184. */ 
  185. return select(selector.replace(rtrim, "$1"), context, results, seed); 

各位朋友,若覺得寫得不錯,幫我頂一下,給點動力,多謝
!

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲天堂av电影| 成人a级免费视频| 精品免费在线视频| 日韩av电影中文字幕| 久久在精品线影院精品国产| 日韩有码片在线观看| 欧美日韩激情视频| 欧美丝袜第一区| 4438全国亚洲精品在线观看视频| 欧美在线欧美在线| 国产精品高潮视频| 亚洲性视频网站| 亲爱的老师9免费观看全集电视剧| 久久久亚洲影院你懂的| 欧美电影免费看| 国产精品一区二区久久| 一区二区国产精品视频| 日韩视频在线一区| 国产不卡av在线| 亚洲国产精彩中文乱码av| 亚洲精品色婷婷福利天堂| 国产精品久久久久久久电影| 亚洲第一福利视频| 日韩高清av在线| 日韩麻豆第一页| 日韩在线免费视频观看| 黑人巨大精品欧美一区免费视频| 日韩在线视频免费观看| 亚洲伊人久久综合| 日本中文字幕不卡免费| 亚洲第一国产精品| 久久久久久久久久久91| 成人网址在线观看| 欧美重口另类videos人妖| 国产精品久久久久久久久借妻| 日韩中文字幕网| 青草成人免费视频| 亚洲男人天堂古典| 久久婷婷国产麻豆91天堂| 欧美另类69精品久久久久9999| 欧美激情国产高清| 欧美性猛交xxxx乱大交| 国产成人一区二区| 福利视频导航一区| 精品国产精品自拍| 亚洲欧美一区二区激情| 日韩女优在线播放| 91在线视频九色| 国产欧美va欧美va香蕉在线| 夜色77av精品影院| 国产成人av网址| 91精品国产网站| 亚洲韩国日本中文字幕| 一区二区三区亚洲| 国产a∨精品一区二区三区不卡| 欧美午夜久久久| 久久久av一区| 日韩视频免费大全中文字幕| 成人激情在线播放| 亚洲最大的免费| 日韩电影在线观看永久视频免费网站| 欧美中文在线观看国产| 国产成人精品在线观看| 97视频在线播放| 欧美激情xxxx性bbbb| 国产精品视频网| 91久久国产精品91久久性色| 久久久精品久久久| 亚洲加勒比久久88色综合| 18一19gay欧美视频网站| 热99久久精品| 久久韩国免费视频| 亚洲视频日韩精品| 欧美精品电影免费在线观看| 欧美午夜片欧美片在线观看| 成人有码在线播放| 国产欧美韩国高清| 欧洲成人免费视频| 精品人伦一区二区三区蜜桃网站| 日韩欧美国产网站| 97碰碰碰免费色视频| 日韩视频在线观看免费| 亚洲专区在线视频| 欧美一级片久久久久久久| 在线观看欧美成人| 欧美高清在线观看| 久久精品视频免费播放| 亚洲精选中文字幕| 中文字幕视频在线免费欧美日韩综合在线看| 国产精品视频久久久久| 国内自拍欧美激情| 亚洲福利视频免费观看| 亚洲精品视频免费| 欧美性猛交xxxx| 亚洲福利在线看| 亚洲aaaaaa| 欧美精品日韩www.p站| 日韩一级裸体免费视频| 日韩精品视频在线观看免费| 人体精品一二三区| 久久精品国产一区二区三区| 在线播放国产一区二区三区| 国产日韩在线看| 日韩精品亚洲精品| 久久成人这里只有精品| 亚洲第一天堂av| 欧美在线一区二区三区四| 亚洲男人av电影| 18一19gay欧美视频网站| 国产视频精品va久久久久久| 色婷婷综合久久久久| 久久久久久这里只有精品| 日韩av色综合| 亚洲999一在线观看www| 九九精品在线播放| 91亚洲国产成人久久精品网站| 丁香五六月婷婷久久激情| 97久久精品人搡人人玩| 亚洲第一二三四五区| 国产99在线|中文| 国内精品久久久久久久久| 热久久免费国产视频| 精品动漫一区二区三区| 欧美一级大片视频| 亚洲成人av在线播放| 亚洲国产欧美一区| 国外成人在线视频| 欧美夫妻性视频| 在线看日韩欧美| 国产日韩综合一区二区性色av| 亚洲精品免费一区二区三区| 精品成人国产在线观看男人呻吟| 亚洲精品在线不卡| 性欧美xxxx交| 高清一区二区三区四区五区| 久久久这里只有精品视频| 精品福利在线观看| 777精品视频| 日韩亚洲欧美成人| 亚洲欧洲在线视频| 久久精品国产亚洲7777| 久久亚洲精品中文字幕冲田杏梨| 日本国产高清不卡| 亚洲情综合五月天| 91在线免费看网站| 黄色成人在线播放| 亚洲视频第一页| 一区二区三区亚洲| 欧美黄色片在线观看| 欧美专区在线视频| 欧美精品制服第一页| 精品视频偷偷看在线观看| 欧美成在线观看| 隔壁老王国产在线精品| 国产欧美精品一区二区三区介绍| 久久久国产在线视频| 国产精品自拍偷拍视频| 亚洲国产欧美精品| 伊人激情综合网| 亚洲欧洲国产伦综合| 精品亚洲va在线va天堂资源站| 亚洲xxx视频| 亚洲国产精品一区二区三区| 国产精品美女久久久久久免费|