1 jQuery 对象Sizzle.selectors.relative中存放了块间关系符和相应的块间关系过滤函数,称为“块间关系过滤函数集”

块间关系符共同拥有4种,其含义和过滤方式如图所看到的。

在函数Sizzle( selector, context, results, seed )从右向左进行过滤时,块间关系过滤函数被

调用,用于检查映射集checkSet中的元素是否匹配块间关系符左側的块表达式。调用时的參

数格式为:

Sizzle.selectors.relative[ 块间关系符 cur  ](  映射集 checkSet,  左側块表达式pop,
contextXML );

块间关系过滤函数接受3个參数:

‰ ‰參数checkSet:映射集,对该元素集合运行过滤操作。

‰ ‰參数part:大多数情况下是块间关系符左側的块表达式,该參数也能够是DOM元素。

‰ ‰參数isXML:布尔值,指示是否执行在一个XML文档中。

块间关系过滤函数实现的3个关键过程例如以下:

1)遍历映射集checkSet。

2)依照块间关系符查找每一个元素的兄弟元素、父元素或祖先元素。

3)检查找到的元素是否匹配參数part,并替换映射集checkSet中相应位置的元素。

a. 假设參数part是标签,则检查找到的元素其节点名称nodeName是否与之相等,

假设相等则替换为找到的元素,不相等则替换为false。

b. 假设參数part是DOM元素,则检查找到的元素是否与之相等,假设相等则替换

为true,不相等则替换为false。

c. 假设參数part是非标签字符串,则调用方法Sizzle.filter( selector, set, inplace, not )过滤。

也就是说,遍历结束后,映射集checkSet中的元素可能会是兄弟元素、父元素、

祖先元素、true或false。



1 块间关系符"+"匹配选择器"prev + next",即匹配全部紧接在元素prev后的兄弟元素

next。比如,$("div + span")、$(".lastdiv + span")。对于从右向左的查找方式,则是检查元

素next之前的兄弟元素是否匹配块表达式prev。

var Expr = Sizzle.selectors = {
relative: {
"+": function(checkSet, part){
//检查參数是否为字符串
var isPartStr = typeof part === "string",
//指示參数part是否为标签字符串
isTag = isPartStr && !rNonWord.test( part ),
//isPartStrNotTag:指示參数part是否是非标签字符串。
isPartStrNotTag = isPartStr && !isTag; if ( isTag ) {
part = part.toLowerCase();
}
/*
遍历映射集checkSet,查找每一个元素的前一个兄弟元素,并替换映
射集checkSet中相应位置的元素,有下面3个逻辑分支:
‰1 假设未找到兄弟元素,则替换为false。
2 ‰假设找到了兄弟元素,而且參数part是标签,则检查兄弟元素的节点名称nodeName
是否与之相等,假设相等则替换为兄弟元素,不相等则替换为false。
3 ‰假设找到了兄弟元素,而且參数part是DOM元素,则检查二者是否相等,假设相等
则替换为true,不相等则替换为false。
因此,在遍历结束后,映射集checkSet中的元素可能会是兄弟元素、true或false。
*/
for ( var i = 0, l = checkSet.length, elem; i < l; i++ ) {
if ( (elem = checkSet[i]) ) {
/*在遍历兄弟元素的同一时候过滤掉非元素节点,而且仅仅要取到一个兄弟元素就
退出while循环。*/
while ( (elem = elem.previousSibling) && elem.nodeType !== 1 ) { } checkSet[i] = isPartStrNotTag || elem && elem.node
Name.toLowerCase() === part ?
elem || false :
elem === part;
}
}
/*
假设參数part是非标签字符串,则调用方法Sizzle.filter( selector,
set, inplace, not )过滤映射集checkSet。对于參数part是标签和DOM元素的情况,在前面遍
历映射集checkSet时已经处理过了。
*/
if ( isPartStrNotTag ) {
Sizzle.filter( part, checkSet, true );
}
},
},
};

2 块间关系符">"用于选择器"parent > child",即匹配父元素parent下的子元素child。

比如,$("div + span")、$(".lastdiv + span")。对于从右向左的查找方式,则是检查子元素

child的父元素是否匹配块表达式parent。

var Expr = Sizzle.selectors = {
relative: {
">": function( checkSet, part ) {
var elem,
isPartStr = typeof part === "string",
i = 0,
l = checkSet.length;
/*
假设參数part是标签,则遍历映射集checkSet,查找每一个元素的
父元素,并检查父元素的节点名称nodeName是否与參数part相等,假设相等则替换映射集
checkSet中相应位置的元素为父元素,不相等则替换为false。
*/
if ( isPartStr && !rNonWord.test( part ) ) {
part = part.toLowerCase(); for ( ; i < l; i++ ) {
elem = checkSet[i]; if ( elem ) {
var parent = elem.parentNode;
checkSet[i] = parent.nodeName.toLowerCase() === part ?
parent : false;
}
}
} else {
/*
假设參数part不是标签,则可能是非标签字符串或DOM元素,同
样遍历映射集checkSet,查找每一个元素的父元素,并替换映射集checkSet中相应位置的元
素,在这个过程中有下面2个逻辑分支:
‰1 ‰假设參数part是非标签字符串,则在遍历映射集checkSet的过程中,替换映射集
checkSet中相应位置的元素为父元素,遍历结束后调用方法Sizzle.filter( selector, set,
inplace, not )过滤映射集checkSet。
‰2 ‰假设參数part是元素,则在遍历映射集checkSet时,检查每一个元素的父元素是否与
之相等,假设相等则替换映射集checkSet中相应位置的元素为true,不相等则替换为
false。
因此,在遍历结束后,映射集checkSet中的元素可能会是父亲元素、true或false。
*/
for ( ; i < l; i++ ) {
elem = checkSet[i]; if ( elem ) {
checkSet[i] = isPartStr ?
elem.parentNode :
elem.parentNode === part;
}
} if ( isPartStr ) {
Sizzle.filter( part, checkSet, true );
}
}
},
},
};

3 块间关系符""用于选择器"ancestor descendant",即匹配祖先元素ancestor的全部后代

元素descendant。比如,$("div button")、$("div .btn")。对于从右向左的查找方式,则是检

查后代元素descendant的祖先元素是否匹配块表达式ancestor。

var Expr = Sizzle.selectors = {
relative: {
"": function(checkSet, part, isXML){
var nodeCheck,
doneName = done++,
checkFn = dirCheck;
/*
‰1 假设參数part是非标签字符串或DOM元素,则调用函数dirCheck()过滤映射集
checkSet。
‰2 假设參数part是标签,则调用函数dirNodeCheck()过滤映射集checkSet。
调用函数dirCheck()和dirNodeCheck()时的參数格式为:
checkFn( 方向 "parentNode/previousSibling", 块表达式 part, 缓存计数器 doneName, 映
射集 checkSet, nodeCheck, isXML )
函数dirCheck()和dirNodeCheck()会遍历映射集checkSet,查找每一个元素的祖先元素,
并检查是否有祖先元素匹配參数part,同一时候替换映射集checkSet中相应位置的元素。
*/
if ( typeof part === "string" && !rNonWord.test( part ) ) {
part = part.toLowerCase();
nodeCheck = part;
checkFn = dirNodeCheck;
} checkFn( "parentNode", part, doneName, checkSet, nodeCheck, isXML );
},
},
};

4 块间关系符"~"用于选择器"prev~siblings",即匹配元素prev之后的全部兄弟元

素siblings。比如,$('div~p')。对于从右向左的查找方式,则是检查元素siblings之前的

兄弟元素是否匹配块表达式prev。

Sizzle.selectors.relative["~"]( checkSet, part )的源代码实现与Sizzle.selectors.relative[""]

( checkSet, part )差点儿一样,两者的差别只在于调用函数dirCheck()和dirNodeCheck()时第

一个參数的值不同,前者是"previousSibling",后者则是"parentNode"。

相关代码例如以下所看到的:

var Expr = Sizzle.selectors = {
relative: {
"~": function( checkSet, part, isXML ) {
var nodeCheck,
doneName = done++,
checkFn = dirCheck; if ( typeof part === "string" && !rNonWord.test( part ) ) {
part = part.toLowerCase();
nodeCheck = part;
checkFn = dirNodeCheck;
} checkFn( "previousSibling", part, doneName, checkSet, nodeCheck,
isXML );
}
},
};

Sizzle.selectors.relative [ 源代码分析 ]的更多相关文章

  1. [转]JQuery - Sizzle选择器引擎原理分析

    原文: https://segmentfault.com/a/1190000003933990 ---------------------------------------------------- ...

  2. Sizzle.filter [ 源代码分析 ]

    最近的研究已Sizzle选择,对于原理中我们也不得不佩服! Sizzle中间filter办法.主要负责元素表达式过滤块的集合,在内部的方法调用Sizzle.selector.fitler滤波操作的操作 ...

  3. 转:RTMPDump源代码分析

    0: 主要函数调用分析 rtmpdump 是一个用来处理 RTMP 流媒体的开源工具包,支持 rtmp://, rtmpt://, rtmpe://, rtmpte://, and rtmps://. ...

  4. Android 中View的绘制机制源代码分析 三

    到眼下为止,measure过程已经解说完了,今天開始我们就来学习layout过程.只是在学习layout过程之前.大家有没有发现我换了编辑器,哈哈.最终下定决心从Html编辑器切换为markdown编 ...

  5. spark(1.1) mllib 源代码分析

    在spark mllib 1.1加入版本stat包,其中包括一些统计数据有关的功能.本文分析中卡方检验和实施的主要原则: 一个.根本 在stat包实现Pierxunka方检验,它包括以下类别 (1)适 ...

  6. LIRe 源代码分析 3:基本接口(ImageSearcher)

    ===================================================== LIRe源代码分析系列文章列表: LIRe 源代码分析 1:整体结构 LIRe 源代码分析 ...

  7. RTMPdump(libRTMP) 源代码分析 9: 接收消息(Message)(接收视音频数据)

    ===================================================== RTMPdump(libRTMP) 源代码分析系列文章: RTMPdump 源代码分析 1: ...

  8. RTMPdump(libRTMP) 源代码分析 6: 建立一个流媒体连接 (NetStream部分 1)

    ===================================================== RTMPdump(libRTMP) 源代码分析系列文章: RTMPdump 源代码分析 1: ...

  9. Cocos2d-x 源代码分析 : Scheduler(定时器) 源代码分析

    源代码版本号 3.1r,转载请注明 我也最终不out了,開始看3.x的源代码了.此时此刻的心情仅仅能是wtf! !!!!!!! !.只是也最终告别CC时代了. cocos2d-x 源代码分析文件夹 h ...

随机推荐

  1. [java面试题]最长的回文字符串中出现确定

    <span style="font-family: Arial, Helvetica, sans-serif;">package com.wzw.util;</s ...

  2. MDCC 2014移动开发者大会 小礼品展商活动

    MDCC 2014移动开发者大会 小礼品展商活动 watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvdGVzdGNzX2Ru/font/5a6L5L2T/fon ...

  3. 【原创】纯OO:从设计到编码写一个FlappyBird (三)

    第二部分请点这里 下面首先来实现Bing接口! 实现Bing接口的类取名SimpleBing. 容易发现,SimpleBing类总的来说要向下,但点击一下又得向上,向上到了一定界限又得向下,但我们又只 ...

  4. MyBatis学习门户网站(一)

    需要jar包:mybatis-3.x.x.jar .假设需要和spring综合,此外,我们需要增加相关的包 1:看到项目文件夹 不要在意红色 2:依照步骤: 1:增加jar包 2:创建数据源(conf ...

  5. 10令人惊叹的模型的影响HTML5应用程序及源代码

    HTML5已经越来越流行起来了.尤其是移动互联网的发展,更是带动了HTML5的迅猛发展,我们也是时候学习HTML5了,以防到时候落伍.今天给大家介绍10款效果惊艳的HTML5应用.方便大家学习,也将应 ...

  6. Android SDK 和 Eclipse ADT 离线安装 教程

    因为google 被限制,就是FQ后,下载的速度依旧非常慢,让人非常崩溃啊,所以这里就分享一下离线安装android SDK 和eclipse ADT 离线安装方法. 安装之前首先已经确保java s ...

  7. CMake入门(二)

    CMake入门(二) 最后更新日期:2014-04-25 by kagula 阅读前提:<CMake入门(一)>.Linux的基本操作 环境: Windows 8.1 64bit英文版.V ...

  8. 【C++探索之旅】第一部分第三课:第一个C++程序

    内容简介 1.第一部分第三课:第一个C++程序 2.第一部分第四课预告:内存的使用 第一个C++程序 经过上两课之后,我们已经知道了什么是编程,编程的语言,编程的必要软件,C++是什么,我们也安装了适 ...

  9. JProgressBar的一个框架

    Frame: package swing.progress; import java.awt.BorderLayout; import java.awt.Frame; import java.awt. ...

  10. 如何从Terminal Command Line编译并运行Scope

    Ubuntu SDK我们大部分的开发者是非常有效的.它甚至可以帮助我们进行在线调试.在这篇文章中,我们介绍了如何使用command line编译和执行我们scope. 1)创建一个主Scope 我们能 ...