jQuery选择器源码解读(四):tokenize方法的Expr.preFilter


Posted in Javascript onMarch 31, 2015

Expr.preFilter是tokenize方法中对ATTR、CHILD、PSEUDO三种选择器进行预处理的方法。具体如下:

Expr.preFilter : {
	"ATTR" : function(match) {
		/*
		 * 完成如下任务:
		 * 1、属性名称解码
		 * 2、属性值解码
		 * 3、若判断符为~=,则在属性值两边加上空格
		 * 4、返回最终的mtach对象
		 * 
		 * match[1]表示属性名称,
		 * match[1].replace(runescape, funescape):将属性名称中的十六进制数解码成
		 *   单字节unicode字符或双字节unicode字符(中文或其它需要两个字节表达的文字)
		 * 正则表达式的详细说明,可以参看我的“详解jQuery选择器正则表达式”文章
		 */
		match[1] = match[1].replace(runescape, funescape);

		/*
		 * 将属性值解码
		 * match[4]:表示放在单引号或双引号内的属性值
		 * match[5]: 表示不用引号括起来的属性值
		 */
		match[3] = (match[4] || match[5] || "").replace(runescape,
				funescape);

		/*
		 * ~=的意思是单词匹配,在W3C中对单词的定义是以空白为不同单词的分隔符
		 * 故此处在match[3]两边加上空格后,可以利用indexOf,正确识别出该单词是否存在
		 */
		if (match[2] === "~=") {
			match[3] = " " + match[3] + " ";
		}

		/*
		 * 返回有用的前四个元素结果
		 */
		return match.slice(0, 4);
	},

	"CHILD" : function(match) {
		/*
		 * 完成如下几项任务:
		 * 1、把命令中child和of-type之前的字符变成小写字符
		 * 2、对于nth开头的选择器检查括号内的数据有效性
		 * 3、match[4]和match[5]分别存放xn+b中的x和b,x和b允许是负数
		 * 4、返回最终的match对象
		 * 
		 * match[1]:(only|first|last|nth|nth-last)中的一个
		 */
		match[1] = match[1].toLowerCase();

		/*
		 * 对于nth-child、nth-of-type、nth-last-child、nth-last-of-type四种类型括号内需设置有效数据
		 * 而其它则括号内不允许有任何数据
		 */
		if (match[1].slice(0, 3) === "nth") {
			/*
			 * 若选择器括号内没有有效参数,则抛出异常
			 * 举例:若选择器是nth或nth(abc)则属于非法选择器
			 */
			if (!match[3]) {
				Sizzle.error(match[0]);
			}
			/*
			 * 下面先以nth-child()为例介绍一下语法,以便更好的理解下面代码的作用
			 * nth-child允许的几种使用方式如下:
			 * 	 :nth-child(even)
			 * 	 :nth-child(odd)
			 * 	 :nth-child(3n)
			 * 	 :nth-child(+2n+1)
			 * 	 :nth-child(2n-1)
			 * 下面代码中赋值号左侧的match[4]、match[5]用于分别记录括号内n前及n后的数值,包括正负号
			 * 对于:nth-child(even)和:nth-child(odd)来说,match[4]为空,
			 *   所以返回 2 * (match[3] === "even" || match[3] === "odd")的计算结果
			 *   因为在js中true=1,false=0,所以(match[3] === "even" || match[3] === "odd")等于1
			 *   因此,2 * (match[3] === "even" || match[3] === "odd")的计算结果为2
			 * 
			 * 等号右侧的“+”的作用是强制类型转换,将之后的字符串转换成数值类型 
			 */
			match[4] = +(match[4] ? match[5] + (match[6] || 1)
					: 2 * (match[3] === "even" || match[3] === "odd"));
			match[5] = +((match[7] + match[8]) || match[3] === "odd");

		} else if (match[3]) {
			/*
			 * 若非nth起头的其它CHILD类型选择器带有括号说明,则抛出异常
			 * 这里jQuery并没有严格按照W3C的规则来判定,因为其允许:first-child()的这种形式存在
			 * 也就是对于jQuery来说:first-child()等同于:first-child,是合法选择器
			 */
			Sizzle.error(match[0]);
		}

		return match;
	},

	"PSEUDO" : function(match) {
		/*
		 * 完成如下任务:
		 * 1、获取伪类中用引号括起来的值
		 * 2、对于非引号括起来的值,若存在伪类嵌套,则进一步解析确定当前伪类实际结束位置,
		 *  获取当前伪类的完整字符串和值
		 * 3、返回match中的前三项的副本。
		 * 
		 * unquoted表示括号内非引号括起来的值,
		 * 以:eq(2)为例,unquoted=2
		 */
		var excess, unquoted = !match[5] && match[2];

		/*
		 * 因为pseudo与child的匹配正则表达式有交集,所以,需要把属于child的部分忽略掉
		 */
		if (matchExpr["CHILD"].test(match[0])) {
			return null;
		}
		/*
		 * 若括号内的值使用引号(match[3])括起来的,
		 * 则将除引号外的值(match[4])赋给match[2]。
		 * match[3]表示引号。
		 */
		if (match[3] && match[4] !== undefined) {
			match[2] = match[4];
		} else if (unquoted
				/*
				 * rpseudo.test(unquoted):用来测试unquoted是否包含伪类,
				 * 若包含伪类,则说明有可能存在伪类嵌套的可能性,需要进一步对unquoted进行解析
				 * 例如: :not(:eq(3))
				 */
				&& rpseudo.test(unquoted)
				&&
				/*
				 * 获取unquoted中连续有效地选择器最后一个字符所在位置
				 */
				(excess = tokenize(unquoted, true))
				&&
				/*
				 * unquoted.indexOf(")", unquoted.length - excess)
				 *   从之前获得的连续有效地选择器最后一个字符所在位置之后找到")"所在位置,
				 *   通常就在当前位置之后。
				 * 再减去unquoted.length,用来获得match[0]中的有效完整的伪类字符串最后位置,
				 *   注意,此时excess是一个负值
				 * 
				 */
				(excess = unquoted.indexOf(")", unquoted.length
						- excess)
						- unquoted.length)) {

			// 获取有效的完整伪类match[0]和伪类括号内的数据match[2]
			match[0] = match[0].slice(0, excess);
			match[2] = unquoted.slice(0, excess);
		}

		// 返回match前三个元素的副本
		return match.slice(0, 3);
	}
}
Javascript 相关文章推荐
缓动函数requestAnimationFrame 更好的实现浏览器经动画
Dec 07 Javascript
jQuery如何取id有.的值一般的方法是取不到的
Apr 18 Javascript
JavaScript使用focus()设置焦点失败的解决方法
Sep 03 Javascript
node.js中的fs.lchmodSync方法使用说明
Dec 16 Javascript
推荐10 款 SVG 动画的 JavaScript 库
Mar 24 Javascript
js实现跨域访问的三种方法
Dec 09 Javascript
深入理解Ajax的get和post请求
Jun 02 Javascript
关于Function中的bind()示例详解
Dec 02 Javascript
原JS实现banner图的常用功能
Jun 12 Javascript
Express + Session 实现登录验证功能
Sep 08 Javascript
vue + axios get下载文件功能
Sep 25 Javascript
vue 路由懒加载中给 Webpack Chunks 命名的方法
Apr 24 Javascript
JavaScript制作简易的微信打飞机
Mar 31 #Javascript
JS获取表格内指定单元格html内容的方法
Mar 31 #Javascript
JS实现为表格动态添加标题的方法
Mar 31 #Javascript
JS实现从表格中动态删除指定行的方法
Mar 31 #Javascript
jQuery选择器源码解读(三):tokenize方法
Mar 31 #Javascript
javascript制作游戏开发碰撞检测的封装代码
Mar 31 #Javascript
jQuery选择器源码解读(二):select方法
Mar 31 #Javascript
You might like
《魔兽争霸3:重制版》更新 多项视觉效果调整
2020/05/04 魔兽争霸
深入解析PHP中逗号与点号的区别
2013/08/05 PHP
php表单敏感字符过滤类
2014/12/08 PHP
php压缩和解压缩字符串的方法
2015/03/14 PHP
PHP中实现crontab代码分享
2015/03/26 PHP
php实现在新浪云中使用imagick生成缩略图并上传的方法
2016/09/26 PHP
PHP实现的登录页面信息提示功能示例
2017/07/24 PHP
js函数使用技巧之 setTimeout(function(){},0)
2009/02/09 Javascript
用jQuery简化JavaScript开发分析
2009/02/19 Javascript
去掉gridPanel表头全选框的小例子
2013/07/18 Javascript
JavaScript学习笔记之Function对象
2015/01/22 Javascript
Javascript for in的缺陷总结
2017/02/03 Javascript
angularjs select 赋值 ng-options配置方法
2018/02/28 Javascript
JavaScript链式调用实例浅析
2018/12/19 Javascript
JavaScript数据结构与算法之基本排序算法定义与效率比较【冒泡、选择、插入排序】
2019/02/21 Javascript
layui 监听select选择 获取当前select的ID名称方法
2019/09/24 Javascript
vue实现输入一位数字转汉字功能
2019/12/13 Javascript
js实现带箭头的进度流程
2020/03/26 Javascript
vue 解决mintui弹窗弹起来,底部页面滚动bug问题
2020/11/12 Javascript
python 将字符串转换成字典dict
2013/03/24 Python
python爬虫实战之爬取京东商城实例教程
2017/04/24 Python
python机器学习案例教程——K最近邻算法的实现
2017/12/28 Python
Django如何实现上传图片功能
2019/08/16 Python
Python Pandas对缺失值的处理方法
2019/09/27 Python
使用Python进行中文繁简转换的实现代码
2019/10/18 Python
python 检查数据中是否有缺失值,删除缺失值的方式
2019/12/02 Python
Python定时器线程池原理详解
2020/02/26 Python
python实现在列表中查找某个元素的下标示例
2020/11/16 Python
公司年会演讲稿范文
2014/01/11 职场文书
搞笑获奖感言
2014/01/30 职场文书
活动策划求职信模板
2014/04/21 职场文书
消防标语大全
2014/06/07 职场文书
学校群众路线专项整治方案
2014/10/31 职场文书
心得体会该怎么写呢?
2019/06/27 职场文书
Python图片处理之图片裁剪教程
2021/05/27 Python
Python移位密码、仿射变换解密实例代码
2021/06/27 Python