读Zepto源码之集合操作
接下来几个篇章,都会解读 zepto 中的跟 dom
相关的方法,也即源码 $.fn
对象中的方法。
读Zepto源码系列文章已经放到了github上,欢迎star: reading-zepto
源码版本
本文阅读的源码为 zepto1.2.0
.forEach()
forEach: emptyArray.forEach
因为 zepto 的 dom
集合是类数组,所以这里只是简单地复制了数组的 forEach
方法。
具体的 forEach
的用法见文档:Array.prototype.forEach()
.reduce()
reduce: emptyArray.reduce
简单地复制了数组的 reduce
方法。
具体的 reduce
的用法见文档:Array.prototype.reduce()
.push()
push: emptyArray.push
简单地复制了数组的 push
方法。
具体的 push
的用法见文档:Array.prototype.push()
.sort()
sort: emptyArray.sort
简单地复制了数组的 sort
方法。
具体的 sort
的用法见文档:Array.prototype.sort()
.splice()
splice: emptyArray.splice
简单地复制了数组的 splice
方法。
具体的 splice
的用法见文档:Array.prototype.splice()
.indexOf()
indexOf: emptyArray.indexOf
简单地复制了数组的 indexOf
方法。
具体的 indexOf
的用法见文档:Array.prototype.indexOf()
.get()
get: function(idx) {
return idx === undefined ? slice.call(this) : this[idx >= 0 ? idx : idx + this.length]
},
这个方法用来获取指定索引值的元素。
不传参(idx === undefined
)时,不传参调用数组的 slice
方法,将集合中的所有元素返回。
当传递的参数大于或等于零(idx
)时,返回相应索引值的元素 this[idx]
,如果为负数,则倒数返回this.[idx + this.length]
。
例如 $('li').get(-1)
返回的是倒数第1个元素,也即最后一个元素
.toArray()
toArray: function() { return this.get() }
toArray
方法是将元素的类数组变成纯数组。toArray
内部不传参调用 get
方法,上面已经分析了,当不传参数时,get
方法调用的是数组方法 slice
, 返回的自然就是纯数组了。
.size()
size: function() {
return this.length
}
size
方法返回的是集合中的 length
属性,也即集合中元素的个数。
.concat()
concat: function() {
var i, value, args = []
for (i = 0; i < arguments.length; i++) {
value = arguments[i]
args[i] = zepto.isZ(value) ? value.toArray() : value
}
return concat.apply(zepto.isZ(this) ? this.toArray() : this, args)
},
数组中也有对应的 concat
方法,为什么不能像上面的方法那样直接调用呢?
这是因为 $.fn
其实是一个类数组对象,并不是真正的数组,如果直接调用 concat
会直接把整个 $.fn
当成数组的一个 item
合并到数组中。
for (i = 0; i < arguments.length; i++) {
value = arguments[i]
args[i] = zepto.isZ(value) ? value.toArray() : value
}
这段是对每个参数进行判断,如果参数是 zepto
的集合(zepto.isZ(value)
),就先调用 toArray
方法,转换成纯数组。
return concat.apply(zepto.isZ(this) ? this.toArray() : this, args)
这段同样对 this
进行了判断,如果为 zepto
集合,也先转换成数组。所以调用 concat
后返回的是纯数组,不再是 zepto
集合。
.map()
map: function(fn) {
return $($.map(this, function(el, i) { return fn.call(el, i, el) }))
}
map
方法的内部调用的是 zepto
的工具函数 $.map
,这在之前已经在《读Zepto源码之工具函数》做过了分析。
return fn.call(el, i, el)
map
方法对回调也做了包装,call
的第一个参数为 el
,因此可以在 map
的回调中通过 this
来拿到每个元素。
map
方法对 $.map
返回的数组调用了 $()
方法,将返回的数组再次包装成 zepto
对象,因此调用 map
方法后得到的数组,同样具有 zepto
集合中的方法。
.slice()
slice: function() {
return $(slice.apply(this, arguments))
}
slice
同样没有直接用数组的原生方法,也像 map
方法一样,将返回的数组再次包装成 zepto
对象。
.each()
each: function(callback) {
emptyArray.every.call(this, function(el, idx) {
return callback.call(el, idx, el) !== false
})
return this
},
zepto
的 each
方法比较巧妙,在方法内部,调用的其实是数组的 every
方法,every
遇到 false
时就会中止遍历,zepto
也正是利用 every
这种特性,让 each
方法也具有了中止遍历的能力,当 callback
返回的值为布尔值 false
时,中止遍历,注意这里用了 !==
,因为 callback
如果没有返回值时,得到的值会是 undefined
,这种情况是需要排除的。
同样,each
的回调中也是可以用 this
拿到每个元素的。
注意,each
方法最后返回的是 this
, 所以在 each
调用完后,还可以继续调用 集合中的其他方法,这就是 zepto
的链式调用,这个跟 map
方法中返回 zepto
集合的原理差不多,只不过 each
返回的是跟原来一样的集合,map
方法返回的是映射后的集合。
.add()
add: function(selector, context) {
return $(uniq(this.concat($(selector, context))))
}
add
可以传递两个参数,selector
和 context
,即选择器和上下文。
add
调用 $(selector, context)
来获取符合条件的集合元素,这在上篇文章《读Zepto源码之神奇的$》已经有详细的论述。
然后调用 concat
方法来合并两个集合,用内部方法 uniq
来过滤掉重复的项,uniq
方法在《读Zepto源码之内部方法》已经有论述。最后也是返回一个 zepto
集合。
系列文章
参考
- Array.prototype.forEach()
- Array.prototype.reduce()
- Array.prototype.push()
- Array.prototype.sort()
- Array.prototype.splice()
- Array.prototype.indexOf()
License
作者:对角另一面
读Zepto源码之集合操作的更多相关文章
- 读 Zepto 源码之集合元素查找
这篇依然是跟 dom 相关的方法,侧重点是跟集合元素查找相关的方法. 读Zepto源码系列文章已经放到了github上,欢迎star: reading-zepto 源码版本 本文阅读的源码为 zept ...
- 读Zepto源码之样式操作
这篇依然是跟 dom 相关的方法,侧重点是操作样式的方法. 读Zepto源码系列文章已经放到了github上,欢迎star: reading-zepto 源码版本 本文阅读的源码为 zepto1.2. ...
- 读Zepto源码之属性操作
这篇依然是跟 dom 相关的方法,侧重点是操作属性的方法. 读Zepto源码系列文章已经放到了github上,欢迎star: reading-zepto 源码版本 本文阅读的源码为 zepto1.2. ...
- 读Zepto源码之操作DOM
这篇依然是跟 dom 相关的方法,侧重点是操作 dom 的方法. 读Zepto源码系列文章已经放到了github上,欢迎star: reading-zepto 源码版本 本文阅读的源码为 zepto1 ...
- 读Zepto源码之Event模块
Event 模块是 Zepto 必备的模块之一,由于对 Event Api 不太熟,Event 对象也比较复杂,所以乍一看 Event 模块的源码,有点懵,细看下去,其实也不太复杂. 读Zepto源码 ...
- 读Zepto源码之Callbacks模块
Callbacks 模块并不是必备的模块,其作用是管理回调函数,为 Defferred 模块提供支持,Defferred 模块又为 Ajax 模块的 promise 风格提供支持,接下来很快就会分析到 ...
- 读Zepto源码之Deferred模块
Deferred 模块也不是必备的模块,但是 ajax 模块中,要用到 promise 风格,必需引入 Deferred 模块.Deferred 也用到了上一篇文章<读Zepto源码之Callb ...
- 读Zepto源码之Ajax模块
Ajax 模块也是经常会用到的模块,Ajax 模块中包含了 jsonp 的现实,和 XMLHttpRequest 的封装. 读 Zepto 源码系列文章已经放到了github上,欢迎star: rea ...
- 读Zepto源码之Selector模块
Selector 模块是对 Zepto 选择器的扩展,使得 Zepto 选择器也可以支持部分 CSS3 选择器和 eq 等 Zepto 定义的选择器. 在阅读本篇文章之前,最好先阅读<读Zept ...
随机推荐
- Luogu1074靶形数独【启发式搜索】
Luogu1074靶形数独 题目描述 小城和小华都是热爱数学的好学生,最近,他们不约而同地迷上了数独游戏,好胜的他 们想用数独来一比高低.但普通的数独对他们来说都过于简单了,于是他们向 Z 博士请教, ...
- 如何使用LightningChart拖放功能进行数据转移 ?
本文主要介绍如何使用LightningChart扩展拖放功能为所有图表组件创建图表,如:系列,标题,轴线等等.支持用鼠标放置自定义对象到另一个图表中,如:可以添加或修改JSON/CSV或其他格式的数据 ...
- VB中的GDI编程-1 设备环境DC
p{ font-size: 15px; } .alexrootdiv>div{ background: #eeeeee; border: 1px solid #aaa; width: 99%; ...
- 看Lucene源码必须知道的基本规则和算法
上中学的时候写作文,最喜欢的季节我都是写冬天.虽然是因为写冬天的人比较少,那时确实也是对其他季节没有什么特殊的偏好,反而一到冬天,自己皮肤会变得特别白.但是冬天啊,看到的只有四季常青盆栽:瓜栗(就是发 ...
- [洛谷P2580]于是他错误的点名开始了
洛谷P2580的一个水题,用啥都能过,不过为了练习一下刚刚学会的字典树,还是认真做一下吧. #include <cstdio> #include <cstring> using ...
- NodeJS+express+mogondb学习笔记01
0.准备工作 安装nodejs环境 官网地址:https://nodejs.org/en/ 下载好了 直接一路安装 也没有什么可以说的 不得不说nodejs对于新手上手还是很友好的,再加上现在n ...
- 【转】JDBC学习笔记(8)——数据库连接池(dbcp&C3P0)
转自:http://www.cnblogs.com/ysw-go/ JDBC数据库连接池的必要性 一.在使用开发基于数据库的web程序时,传统的模式基本是按一下步骤: 1)在主程序(如servlet/ ...
- Linux防火墙配置—访问外网WEB
一.实验目标 1.本次实验在"Linux基础网络搭建实验"的基础上,在外网虚拟机上搭建WEB服务,并分别配置外网和网关的防火墙规则,使内网能够访问WEB服务 2.Linux基础网络 ...
- ActiveMQ学习心得:连接池的简单实现和模板模式的应用
一.安装activemq 下载地址:https://archive.apache.org/dist/activemq/5.13.0/apache-activemq-5.13.0-bin.zip 下载完 ...
- 2017-4-24 WinForm开发基础、窗体的属性CenterScreen
WinForm中文名称: Windows窗体,是·Net开发平台中对Windows Form的一种称谓. 客户端应用程序:C/S 客户端很重要的特点:可以操作用户电脑上的文件 窗体属性:窗体种类: + ...