最近的研究已Sizzle选择,对于原理中我们也不得不佩服!

Sizzle中间filter办法。主要负责元素表达式过滤块的集合,在内部的方法调用Sizzle.selector.fitler滤波操作的操作方法。

Sizzle.filter要点5:

1 使用LeftMatch确定表达式类型。

2 调用Sizzle.selectors.preFilter预过虑函数,运行过滤前的修正。

3 调用Sizzle.selectors.filter[ type ] 中相应的过滤函数,运行过滤操作,假设返回false,刚将元素集合中的相应位置的元素替换为false;

4 删除表达式中已过滤的部分

5 反复1 - 4 步骤

源代码例如以下:

Sizzle.filter = function( expr, set, inplace, not ) {
//expr 块表达式
//set 待过滤的元素集合
//inplace 是否替换,假设为true,那么当set中的元素与expr不匹配时,刚将不匹配的元素替换为false,
//否则。构造新的数组,仅仅保留匹配的元素
//not: 假设为true,则去掉匹配元素,保留不匹配元素。假设为false,则保留匹配元素,去掉不匹配元素
var match, anyFound,
type, found, item, filter, left,
i, pass,
old = expr,
result = [],
curLoop = set,
isXMLFilter = set && set[0] && Sizzle.isXML( set[0] );
//while 循环,无限循环用expr过滤set元素,直到expr为空,由于在最后会将已经匹配过的部分删除,所以expr会越来越短
while ( expr && set.length ) {
for ( type in Expr.filter ) {
//这里用来确定表达式类型。依据leftMatch正则
//这里是match正则,LeftMatch仅仅是要该正则加上了前缀和后缀
/*
match: {
ID: /#((?:[\w\u00c0-\uFFFF\-]|\\.)+)/,
CLASS: /\.((?:[\w\u00c0-\uFFFF\-]|\\.)+)/,
NAME: /\[name=['"]*((?:[\w\u00c0-\uFFFF\-]|\\.)+)['"]*\]/,
ATTR: /\[\s*((?:[\w\u00c0-\uFFFF\-]|\\.)+)\s*(?:(\S? =)\s*(?:(['"])(.*? )\3|(#?(? :[\w\u00c0-\uFFFF\-]|\\.)*)|)|)\s*\]/,
TAG: /^((? :[\w\u00c0-\uFFFF\*\-]|\\.)+)/,
CHILD: /:(only|nth|last|first)-child(?:\(\s*(even|odd|(? :[+\-]?\d+|(? :[+\-]?\d*)? n\s*(?:[+\-]\s*\d+)?))\s*\))?/,
POS: /:(nth|eq|gt|lt|first|last|even|odd)(?:\((\d*)\))?(?=[^\-]|$)/,
PSEUDO: /:((? :[\w\u00c0-\uFFFF\-]|\\.)+)(?:\((['"]?)((?:\([^\)]+\)|[^\(\)]*)+)\2\))?/
},
*/ if ( (match = Expr.leftMatch[ type ].exec( expr )) != null && match[2] ) {
filter = Expr.filter[ type ];
//确定前缀
left = match[1]; anyFound = false; match.splice(1,1);
//这里官方说是匹配 \\ 假设是 \\ 那么表示后面的元素都是被转义,刚不须要再去过滤,
//只是个人觉得这个地方似乎无效的 left.subster(left.length - 1)
//这个字符的长度仅仅会是1 怎么会和 \\ 相等呢
if ( left.substr( left.length - 1 ) === "\\" ) {
continue;
}
//重置result为空数组,用于缩小候选集,result用来存放通过过滤的元素
if ( curLoop === result ) {
result = [];
} if ( Expr.preFilter[ type ] ) {
//这里会进行预过滤操作,主要对表达式进行改动正,如转义\\,取出空格等
//针对不同和类型,会有不同的方式,详情能够见preFilter方法
match = Expr.preFilter[ type ]( match, curLoop, inplace, result, not, isXMLFilter );
/*match:
1 true: 继续须要运行过滤。尚不到运行过虑函数的时候,直接运行continue,不再去调用filter方法。如pos,child
2 false: 已经运行了过滤,缩小了候选集如:CLASS
3 字符串: 修正之后的过滤參数。后面会继续调用相应的过虑函数
*/ if ( !match ) {
anyFound = found = true; } else if ( match === true ) {
continue;
}
}
//返回的是修正的过滤參数,刚继续运行
if ( match ) {
//遍历候选集中的元素
for ( i = 0; (item = curLoop[i]) != null; i++ ) {
//元素存在
if ( item ) {
//对其进行过滤,返回结果
found = filter( item, match, i, curLoop );
//这里与 not 进行 异或 操作
pass = not ^ found;
//为替换模式,而且found 不为空
if ( inplace && found != null ) {
//假设通过。那么为true
if ( pass ) {
anyFound = true;
} else {
//否则将元素替换为false
curLoop[i] = false;
} } else if ( pass ) {
//这要须要注意的是既然pass为真了。 那么就不须要推断了found,由于pass为found与not异或的结果
//假设通过了,而且为非替换模式,那么将元素放入到新的数组result中
result.push( item );
anyFound = true;
}
}
}
}
//found不为undefined
if ( found !== undefined ) {
if ( !inplace ) {
//非替换模式,将结果复制给curLoop,
curLoop = result;
}
//删除已经过滤过了的部分表达式
expr = expr.replace( Expr.match[ type ], "" );
//不匹配,返回空
if ( !anyFound ) {
return [];
}
//这里的break,说明已经进行过一次过虑,已经找到相应的type,能够选择跳出当前type的循环,
break;
}
}
} // 这里主要是匹配,假设最后过滤的表达式没有变化。那么觉得过滤表达式有问题
if ( expr === old ) {
if ( anyFound == null ) {
Sizzle.error( expr ); } else {
break;
}
}
//备份expr,继续循环
old = expr;
}
//返回过滤的结水果
return curLoop;
};

版权声明:本文博主原创文章。转载声明转载地址。

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

  1. LAV Filter 源代码分析 4: LAV Video (2)

    上一篇文章分析了LAV Filter 中的LAV Video的两个主要的类:CLAVVideo和CDecodeThread.文章:LAV Filter 源代码分析 3: LAV Video (1) 在 ...

  2. 转:LAV Filter 源代码分析

    1: 总体结构 LAV Filter 是一款视频分离和解码软件,他的分离器封装了FFMPEG中的libavformat,解码器则封装了FFMPEG中的libavcodec.它支持十分广泛的视音频格式. ...

  3. JQuery Sizzle引擎源代码分析

    最近在拜读艾伦在慕课网上写的JQuery课程,感觉在国内对JQuery代码分析透彻的人没几个能比得过艾伦.有没有吹牛?是不是我说大话了? 什么是Sizzle引擎? 我们经常使用JQuery的选择器查询 ...

  4. LAV Filter 源代码分析 3: LAV Video (1)

    LAV Video 是使用很广泛的DirectShow Filter.它封装了FFMPEG中的libavcodec,支持十分广泛的视频格式的解码.在这里对其源代码进行详细的分析. LAV Video ...

  5. Sizzle.selectors.relative [ 源代码分析 ]

    1 jQuery 对象Sizzle.selectors.relative中存放了块间关系符和相应的块间关系过滤函数,称为"块间关系过滤函数集" 块间关系符共同拥有4种,其含义和过滤 ...

  6. ffdshow 源代码分析 4: 位图覆盖滤镜(滤镜部分Filter)

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

  7. 转:SDL2源代码分析

    1:初始化(SDL_Init()) SDL简介 有关SDL的简介在<最简单的视音频播放示例7:SDL2播放RGB/YUV>以及<最简单的视音频播放示例9:SDL2播放PCM>中 ...

  8. 转:RTMPDump源代码分析

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

  9. 转:ffdshow 源代码分析

    ffdshow神奇的功能:视频播放时显示运动矢量和QP FFDShow可以称得上是全能的解码.编码器.最初FFDShow只是mpeg视频解码器,不过现在他能做到的远不止于此.它能够解码的视频格式已经远 ...

随机推荐

  1. SQL Server 字段类型 decimal(18,6)小数点前是几位?记一次数据库SP的BUG处理

    原文:SQL Server 字段类型 decimal(18,6)小数点前是几位?记一次数据库SP的BUG处理 SQL Server 字段类型 decimal(18,6)小数点前是几位? 不可否认,这是 ...

  2. 【转】C# String.Format数字格式化输出各种转换{0:N2} {0:D2} {0:C2}...

    ; //格式为sring输出 // Label1.Text = string.Format("asdfadsf{0}adsfasdf",a); // Label2.Text = & ...

  3. [Oracle] 接线表

    于OLTP制,嵌套连接占70%左右,哈希联接占20%,合并排序连接帐户10%. 嵌套连接 算法:嵌套连接从两个表分选出小表为驱动表,大表为被驱动表.先訪问驱动表(仅仅訪问1次).然后依据驱动表返回的行 ...

  4. CSDN-markdown语法之怎样使用LaTeX语法编写数学公式

    文件夹 文件夹 正文 标记公式 行内公式 块级公式 上标和下标 分数表示 各种括号 根号表示 省略号 矢量表示 间隔空间 希腊字母 特殊字符 关系运算符 集合运算符 对数运算符 三角运算符 微积分运算 ...

  5. python学习笔记之四:条件,循环和其他语句

    前面已经介绍过几种基本语句(print,import,赋值语句),下面我们来介绍条件语句,循环语句. 一. print和import的更多信息 1.1 使用逗号输出 A.打印多个表达式,用逗号隔开,会 ...

  6. Android编程心得-Service数据绑定初步

    在Android里,Service的数据绑定是一种重要的用法,我们知道Service与Activity一样是运行在当前应用进程的主线程里面的,他们之间交互的方式有多种,下面我来介绍一下如何使用数据绑定 ...

  7. php laravel 帧 该文件上传

    好,我承认我的忠告. 今天laravel框架编写一个文件上传部分.总能找到不正确的路径.但是,终于攻克. 以下我分享一下自己的学习体会吧. client <form method="P ...

  8. gitLab添加ssh key

    电脑新装了一台虚拟机,想要和gitLab建立一个安全的ssh连接,步骤如下 1.本机生成ssh key 系统环境:Linux 使用root用户登录,执行命令:ssh-keygen -t rsa -C ...

  9. poj 3259(bellman最短路径)

    Wormholes Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 30169   Accepted: 10914 Descr ...

  10. Android Tombstone/Crash的log分析和定位

    有一句话叫做常在河边走,哪有不湿鞋.我们这些研究和开发Android的project师正应了这句话,相必大家在调试的时候常常会遇到这么个东西吧 *** *** *** *** *** *** *** ...