摘要:返回值为,如果能查找到元素,则将元素以数组的形式返回,否则返回空数组排除不合法的。的第一个字符为,并且为标签。如果存在,则查找下选择器为的所有子元素。正则表达式为如果没有指定标签名,则获取标签名。包裹元素的即为所需要获取的。
经过前面三章的铺垫,这篇终于写到了戏肉。在用 zepto 时,肯定离不开这个神奇的 $ 符号,这篇文章将会看看 zepto 是如何实现 $ 的。
读Zepto源码系列文章已经放到了github上,欢迎star: reading-zepto
源码版本本文阅读的源码为 zepto1.2.0
zepto的css选择器 zepto.qsa我们都知道,很多时候,我们都用$ 来获取DOM对象,这跟 zepto.qsa 有很大的关系。
源码zepto.qsa = function(element, selector) { var found, // 已经找的到DOM maybeID = selector[0] == "#", // 是否为ID maybeClass = !maybeID && selector[0] == ".", // 是否为class nameOnly = maybeID || maybeClass ? selector.slice(1) : selector, // 将id或class前面的符号去掉 isSimple = simpleSelectorRE.test(nameOnly) // 是否为单个选择器 return (element.getElementById && isSimple && maybeID) ? ((found = element.getElementById(nameOnly)) ? [found] : []) : (element.nodeType !== 1 && element.nodeType !== 9 && element.nodeType !== 11) ? [] : slice.call( isSimple && !maybeID && element.getElementsByClassName ? maybeClass ? element.getElementsByClassName(nameOnly) : element.getElementsByTagName(selector) : element.querySelectorAll(selector) ) }
以上是 qsa 的所有代码,里面有用到一个正则表达式 simpleSelectorRE,先将这个正则消化下。
simpleSelectorRE = /^[w-]*$/,
看到这个正则其实是匹配 a-z、A-Z、0-9、下划线、连词符 组合起来的单词,这其实就是单个 id 和 class 的命名规则。
从 return 中可以看出,qsa 其实是根据不同情况分别调用了原生的 getElementById、getElementsByClassName 、getElementsByTagName 和 querySelectorAll 的方法。
为什么要这么麻烦,不直接调用 querySelectorAll 方法呢?这是出于性能的考虑。这里有个简单的测试。这个测试里,页面上只有一个元素,如果比较复杂的时候,差距更加明显。
好了,开始逐行分析代码。
参数element 开始查找的元素
selector 选择器
变量found: 已经找到的元素
maybeID = selector[0] == "#": 判断选择器的第一个字符是否为 #, 如果是 # ,则可能是 id 选择器
maybeClass = !maybeID && selector[0] == "." 如果不是 id 选择器,并且选择器的第一个字符为 . ,则可能是 class 选择器
nameOnly = maybeID || maybeClass ? selector.slice(1) : selector ,如果为 id 选择器或者 class 选择器,则将第一个字符去掉
isSimple = simpleSelectorRE.test(nameOnly) 是否为单选择器,即 .single 的形式,不是 .first .secend 等形式
element.getElementById(element.getElementById && isSimple && maybeID) 这是采用 element.getElementById 的条件。
首先要确保 element 具有 getElementById 的方法。getElementById 的方法是在 document 上的,Chrome等浏览器上,element 可能并不具有 geElementById 的方法,具体可以看看这篇文章:各浏览器对document.getElementById等方法的实现差异解析
然后要确保选择器为单选择器,并且为 id 选择器。
返回值为 ((found = element.getElementById(nameOnly)) ? [found] : []), 如果能查找到元素,则将元素以数组的形式返回,否则返回空数组
排除不合法的elementelement.nodeType !== 1 && element.nodeType !== 9 && element.nodeType !== 11 。1 对应的是 Node.ELEMENT_NODE ,9 对应的是 Node.DOCUMENT_NODE , 11 对应的是 Node.DOCUMENT_FRAGMENT_NODE ,如果不为以上三种类型,直接返回 []。
终极三元表达式slice.call( isSimple && !maybeID && element.getElementsByClassName ? // 如果为单选择器并且不为id选择器并且存在getElementsByClassName方法,进入下一个三元表达式判断 maybeClass ? element.getElementsByClassName(nameOnly) : // 如果为class选择器,则采用getElementsByClassName element.getElementsByTagName(selector) : // 否则采用getElementsByTagName方法 element.querySelectorAll(selector) // 以上情况都不是,则用querySelectorAll )
这里用了 slice.call 处理所获取到的集合,这样,获取到的DOM集合就可以直接使用数组的方法了。
zepto.Z 函数从第一篇代码结构中我们已经知道,其实实现 $ 函数的核心是 zepto.init ,而 zepto.init 最终返回的是 zepto.Z 的结果。那就先来看看 zepto.Z
zepto.Z = function(dom, selector) { return new Z(dom, selector) }
zepto.Z 的代码很简单,返回的是 Z 函数的实例。那接下来再看看 Z 函数:
function Z(dom, selector) { var i, len = dom ? dom.length : 0 for (i = 0; i < len; i++) this[i] = dom[i] this.length = len this.selector = selector || "" }
Z 函数做的事情也很简单,就是将 dom 数组转化为类数组的形式,并设置对应的 length 属性和 selector 属性。
zepto.isZzepto.isZ = function(object) { return object instanceof zepto.Z }
既然看了 Z 函数,就顺便也将 isZ 也一起看了吧。isZ 函数用来判断参数 object 是否为 Z 的实例,这在 init 中会用到。
$的实现 zepto.init 函数 $的实现$ = function(selector, context) { return zepto.init(selector, context) }
可以看到,其实 $ 调用的就是 zepto.init 这个内部方法。
zepto.initzepto.init = function(selector, context) { var dom // dom 集合 if (!selector) return zepto.Z() // 分支1 else if (typeof selector == "string") { // 分支2 selector = selector.trim() if (selector[0] == "<" && fragmentRE.test(selector)) dom = zepto.fragment(selector, RegExp.$1, context), selector = null else if (context !== undefined) return $(context).find(selector) else dom = zepto.qsa(document, selector) } else if (isFunction(selector)) return $(document).ready(selector) // 分支3 else if (zepto.isZ(selector)) return selector // 分支4 else { // 分支5 if (isArray(selector)) dom = compact(selector) else if (isObject(selector)) dom = [selector], selector = null else if (fragmentRE.test(selector)) dom = zepto.fragment(selector.trim(), RegExp.$1, context), selector = null else if (context !== undefined) return $(context).find(selector) else dom = zepto.qsa(document, selector) } return zepto.Z(dom, selector) }
这个 init 方法代码量不多,但是有大量的 if else, 希望我可以说得清楚
$的用法$(selector, [context]) ⇒ collection // 用法1 $(不传参调用) ⇒ same collection // 用法2 $( ) ⇒ collection // 用法3 $(htmlString) ⇒ collection // 用法4 $(htmlString, attributes) ⇒ collection v1.0+ // 用法5 Zepto(function($){ ... }) // 用法6
直接调用 $() 时,对应的是分支1的情况: if (!selector) return zepto.Z() ,返回的是空的 Z 对象
selector 为 String 时当 selector 为 string 时,对应的代码在分支2,对应的用法是用法1、用法4和用法5
在这个分支里,又有三个子分支。一一来看一下:
第一个的判断条件为 selector[0] == "<" && fragmentRE.test(selector) 。selector 的第一个字符为 < ,并且为html标签 。fragmentRE 的定义如下 fragmentRE = /^s*<(w+|!)[^>]*>/ ,这个其实就是用来判断字符串是否为标签。 我对正则也不太熟,这里就不再展开。
如果满足条件,则执行如下代码:dom = zepto.fragment(selector, RegExp.$1, context), selector = null。 zepto.fragment 其实是通过 htmlString 返回一个dom集合。这个函数稍后会说到,这里先不展开。这里对应的是用法4和用法5。
如果不满足第一个判断条件,则再判断 context !== undefined (上下文是否存在)。如果存在,则查找 context 下选择器为 selector 的所有子元素: $(context).find(selector) 。这个分支对应的是用法1
否则,调用 zepto.qsa 方法,查找 document 下的所有 selector : dom = zepto.qsa(document, selector)。这里对应的是用法1。
selector 为 Function 时对应的代码在分支3,对应的用法是用法6
这个分支很简单,在页面加载完毕后,再执行回调方法:$(document).ready(selector)
用过 zepto 的应该都熟悉这种用法: $(function() {})。其实走的就是这个分支
selector 为 Z 对象时对应的代码在分支4,对应的用法是用法2
如果参数已经为 Z 对象(zepto.isZ(selector)),则不需要做任何事情,直接原对象返回就可以了。
selector 为其他情况如果为数组时(isArray(selector)), 将数组展平(dom = compact(selector))
如果为对象时(isObject(selector)),将对象包裹成数组(dom = [selector])。
以上两种情况对应的是用法3,将dom对象或dom集合转化为 z 对象
如果为标签(fragmentRE.test(selector)),执行跟分支1一模一样的代码。这里判断在上面已经做过了,为什么要再来一次呢?我也不太明白,有明白的可以跟我说下。
经过一轮又一轮的判断和 selector 重置,现在终于可以调用 z 函数了: zepto.Z(dom, selector) ,init 的最后,将收集到的 dom 集合和对应的 selector 传入 Z 函数,返回 Z 对象。
zepto.fragmentzepto.fragment = function(html, name, properties) { var dom, nodes, container if (singleTagRE.test(html)) dom = $(document.createElement(RegExp.$1)) if (!dom) { if (html.replace) html = html.replace(tagExpanderRE, "<$1>$2>") if (name === undefined) name = fragmentRE.test(html) && RegExp.$1 if (!(name in containers)) name = "*" container = containers[name] container.innerHTML = "" + html dom = $.each(slice.call(container.childNodes), function() { container.removeChild(this) }) } if (isPlainObject(properties)) { nodes = $(dom) $.each(properties, function(key, value) { if (methodAttributes.indexOf(key) > -1) nodes[key](value) else nodes.attr(key, value) }) } return dom }
fragment 的作用的是将html片断转换成dom数组形式。
首先判断是否为标签的形式 singleTagRE.test(html) (如), 如果是,则采用该标签名来创建dom对象 dom = $(document.createElement(RegExp.$1)),不用再作其他处理。singleTagRE = /^<(w+)s*/?>(?:1>|)$/。
如果尚未获取到 dom,接着进行:
if (html.replace) html = html.replace(tagExpanderRE, "<$1>$2>")
这段是对 html 进行修复,如
修复成 。正则表达式为 tagExpanderRE = /<(?!area|br|col|embed|hr|img|input|link|meta|param)(([w:]+)[^>]*)/>/igif (name === undefined) name = fragmentRE.test(html) && RegExp.$1
如果没有指定标签名,则获取标签名。如传入
if (!(name in containers)) name = "*" container = containers[name] container.innerHTML = "" + html dom = $.each(slice.call(container.childNodes), function() { container.removeChild(this) }) } // containers 已经开头定义,如下 table = document.createElement("table"), tableRow = document.createElement("tr"), containers = { "tr": document.createElement("tbody"), "tbody": table, "thead": table, "tfoot": table, "td": tableRow, "th": tableRow, "*": document.createElement("div") }
检测 name 是否为特殊的元素,如 tr 要用 tbody 包裹,其他的元素用 div 包裹。包裹元素的 childNodes 即为所需要获取的 dom 。
if (isPlainObject(properties)) { nodes = $(dom) $.each(properties, function(key, value) { if (methodAttributes.indexOf(key) > -1) nodes[key](value) else nodes.attr(key, value) }) } // methodAttributes 在上面已经定义,定义如下 methodAttributes = ["val", "css", "html", "text", "data", "width", "height", "offset"]
如果属性值为纯对象,则给元素设置属性。
如果所需设置的属性,zepto已经定义了相应的方法,则调用zepto对应的方法,否则统一调用zepto的attr 方法设置属性。
最后将 dom 返回
系列文章读Zepto源码之代码结构
读 Zepto 源码之内部方法
读Zepto源码之工具函数
参考各浏览器对document.getElementById等方法的实现差异解析
Node.nodeType
最后,所有文章都会同步发送到微信公众号上,欢迎关注,欢迎提意见:
作者:对角另一面
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/82996.html
摘要:如果伪类的参数不可以用转换,则参数为字符串,用正则将字符串前后的或去掉,再赋值给最后执行回调,将解释出来的参数传入回调函数中,将执行结果返回。重写的方法,改过的调用的是方法,在回调函数中处理大部分逻辑。 Selector 模块是对 Zepto 选择器的扩展,使得 Zepto 选择器也可以支持部分 CSS3 选择器和 eq 等 Zepto 定义的选择器。 在阅读本篇文章之前,最好先阅读《...
摘要:调用来获取符合条件的集合元素,这在上篇文章读源码之神奇的已经有详细的论述。然后调用方法来合并两个集合,用内部方法来过滤掉重复的项,方法在读源码之内部方法已经有论述。最后也是返回一个集合。 接下来几个篇章,都会解读 zepto 中的跟 dom 相关的方法,也即源码 $.fn 对象中的方法。 读Zepto源码系列文章已经放到了github上,欢迎star: reading-zepto 源码...
摘要:读源码系列文章已经放到了上,欢迎源码版本本文阅读的源码为改写原有的方法模块改写了以上这些方法,这些方法在调用的时候,会为返回的结果添加的属性,用来保存原来的集合。方法的分析可以看读源码之模块。 Stack 模块为 Zepto 添加了 addSelf 和 end 方法。 读 Zepto 源码系列文章已经放到了github上,欢迎star: reading-zepto 源码版本 本文阅读的...
摘要:模块是为解决移动版加载图片过大过多时崩溃的问题。因为没有处理过这样的场景,所以这部分的代码解释不会太多,为了说明这个问题,我翻译了这篇文章作为附文怎样处理移动端对图片资源的限制,更详细地解释了这个模块的应用场景。 assets 模块是为解决 Safari 移动版加载图片过大过多时崩溃的问题。因为没有处理过这样的场景,所以这部分的代码解释不会太多,为了说明这个问题,我翻译了《How to...
摘要:模块基于上的事件的封装,利用属性,封装出系列事件。这个判断需要引入设备侦测模块。然后是监测事件,根据这三个事件,可以组合出和事件。其中变量对象和模块中的对象的作用差不多,可以先看看读源码之模块对模块的分析。 Gesture 模块基于 IOS 上的 Gesture 事件的封装,利用 scale 属性,封装出 pinch 系列事件。 读 Zepto 源码系列文章已经放到了github上,欢...
阅读 577·2023-04-25 16:00
阅读 1623·2019-08-26 13:54
阅读 2502·2019-08-26 13:47
阅读 3432·2019-08-26 13:39
阅读 1050·2019-08-26 13:37
阅读 2745·2019-08-26 10:21
阅读 3542·2019-08-23 18:19
阅读 1607·2019-08-23 18:02