读 zepto 源码之工具函数
2017-05-15 15:03
330 查看
4000
本文阅读的源码为 zepto1.2.0
所以我们先看看内部方法
如果
如果
3.1. 如果
3.2. 如果
3.3. 将
现在,再看看
在说原理之前,先来看看
在
后面可以有多个
在
然后判断第一个参数
最后就比较简单了,循环源对象数组
分别调用
先来看看调用方式:
如果
注意回调函数调用了
在遍历的时候,还对回调函数的返回值进行判断,如果回调函数返回
可以遍历数组(类数组)或对象中的元素,根据回调函数的返回值,将返回值组成一个新的数组,并将该数组扁平化后返回,会将
先来看看调用方式:
该方法是将字符串转换成驼峰式的字符串
用来检查给定的父节点中是否包含有给定的子节点,源码如下:
先来看看调用:
参数
用来检测浏览器是否支持
否则,返回另一外匿名函数。该函数会一直向上寻找
否则返回
该函数其实就是数组的
从源码中也可以看出,
返回指定元素在数组中的索引值
先来看看调用
第一个参数
第三个参数
判断是否为数组
判读是否为函数
是否为数值
判断是否为数值,需要满足以下条件
不为
不为布尔值
不为NaN(当传进来的参数不为数值或如
为有限数值
当传进来的参数为字符串的形式,如
是否为纯粹对象,即以
是否为浏览器的
空函数
这个在需要传递回调函数作为参数,但是又不想在回调函数中做任何事情的时候会非常有用,这时,只需要传递一个空函数即可。
将标准JSON格式的字符串解释成JSON
其实就是调用原生的
并且在浏览器不支持的情况下,
删除字符串头尾的空格
如果参数为
类型检测
源码版本
本文阅读的源码为 zepto1.2.0
$.extend
$.extend方法可以用来扩展目标对象的属性。目标对象的同名属性会被源对象的属性覆盖。
$.extend其实调用的是内部方法
extend,
所以我们先看看内部方法
extend的具体实现。
function extend(target, source, deep) { for (key in source) // 遍历源对象的属性值 if (deep && (isPlainObject(source[key]) || isArray(source[key]))) { // 如果为深度复制,并且源对象的属性值为纯粹对象或者数组 if (isPlainObject(source[key]) && !isPlainObject(target[key])) // 如果为纯粹对象 target[key] = {} // 如果源对象的属性值为纯粹对象,并且目标对象对应的属性值不为纯粹对象,则将目标对象对应的属性值置为空对象 if (isArray(source[key]) && !isArray(target[key])) // 如果源对象的属性值为数组,并且目标对象对应的属性值不为数组,则将目标对象对应的属性值置为空数组 target[key] = [] extend(target[key], source[key], deep) // 递归调用extend函数 } else if (source[key] !== undefined) target[key] = source[key] // 不对undefined值进行复制 }
extend的第一个参数
taget为目标对象,
source为源对象,
deep表示是否为深度复制。当
deep为
true时为深度复制,
false时为浅复制。
extend函数用
for···in对
source的属性进行遍历
如果
deep为
false时,只进行浅复制,将
source中不为
undefined的值赋值到
target对应的属性中(注意,这里用的是
!==,不是
!=,所以只排除严格为
undefined的值,不包含
null)。如果
source对应的属性值为对象或者数组,会保持该对象或数组的引用。
如果
deep为
true,并且
source的属性值为纯粹对象或者数组时
3.1. 如果
source的属性为纯粹对象,并且
target对应的属性不为纯粹对象时,将
target的对应属性设置为空对象
3.2. 如果
source的属性为数组,并且
target对应属性不为数组时,将
target的对应属性设置为空数组
3.3. 将
source和
target对应的属性及
deep作为参数,递归调用
extend函数,以实现深度复制。
现在,再看看
$.extend的具体实现
$.extend = function(target) { var deep, args = slice.call(arguments, 1) if (typeof target == 'boolean') { deep = target target = args.shift() } args.forEach(function(arg) { extend(target, arg, deep) }) return target }
在说原理之前,先来看看
$.extend的调用方式,调用方式如下:
$.extend(target, [source, [source2, ...]]) 或 $.extend(true, target, [source, ...])
在
$.extend中,如果不需要深度复制,第一个参数可以是目标对象
target,
后面可以有多个
source源对象。如果需要深度复制,第一个参数为
deep,第二个参数为
target,为目标对象,后面可以有多个
source源对象。
$.extend函数的参数设计得很优雅,不需要深度复制时,可以不用显式地将
deep置为
false。这是如何做到的呢?
在
$.extend函数中,定义了一个数组
args,用来接受除第一个参数外的所有参数。
然后判断第一个参数
target是否为布尔值,如果为布尔值,表示第一个参数为
deep,那么第二个才为目标对象,因此需要重新为
target赋值为
args.shift()。
最后就比较简单了,循环源对象数组
args,
分别调用
extend方法,实现对目标对象的扩展。
$.each
$.each用来遍历数组或者对象,源码如下:
$.each = function(elements, callback) { var i, key if (likeArray(elements)) { // 类数组 for (i = 0; i < elements.length; i++) if (callback.call(elements[i], i, elements[i]) === false) return elements } else { // 对象 for (key in elements) if (callback.call(elements[key], key, elements[key]) === false) return elements } return elements }
先来看看调用方式:
$.each(collection, function(index, item){ ... })
$.each接收两个参数,第一个参数
elements为需要遍历的数组或者对象,第二个
callback为回调函数。
如果
elements为数组,用
for循环,调用
callback,并且将数组索引
index和元素值
item传给回调函数作为参数;如果为对象,用
for···in遍历属性值,并且将属性
key及属性值传给回调函数作为参数。
注意回调函数调用了
call方法,
call的第一个参数为当前元素值或当前属性值,所以回调函数的上下文变成了当前元素值或属性值,也就是说回调函数中的
this指向的是
item。这在dom集合的遍历中相当有用。
在遍历的时候,还对回调函数的返回值进行判断,如果回调函数返回
false(
if (callback.call(elements[i], i, elements[i]) === false)) ,立即中断遍历。
$.each调用结束后,会将遍历的数组或对象(
elements)返回。
$.map
可以遍历数组(类数组)或对象中的元素,根据回调函数的返回值,将返回值组成一个新的数组,并将该数组扁平化后返回,会将 null及
undefined排除。
$.map = function(elements, callback) { var value, values = [], i, key if (likeArray(elements)) for (i = 0; i < elements.length; i++) { value = callback(elements[i], i) if (value != null) values.push(value) } else for (key in elements) { value = callback(elements[key], key) if (value != null) values.push(value) } return flatten(values) }
先来看看调用方式:
$.map(collection, function(item, index){ ... })
elements为类数组或者对象。
callback为回调函数。当为类数组时,用
for循环,当为对象时,用
for···in循环。并且将对应的元素(属性值)及索引(属性名)传递给回调函数,如果回调函数的返回值不为
null或者
undefined,则将返回值存入新数组中,最后将新数组扁平化后返回。
$.camelCase
该方法是将字符串转换成驼峰式的字符串$.camelCase = camelize
$.camelCase调用的是内部方法
camelize,该方法在前一篇文章《读Zepto源码之内部方法》中已有阐述,本篇文章就不再展开。
$.contains
用来检查给定的父节点中是否包含有给定的子节点,源码如下:$.contains = document.documentElement.contains ? function(parent, node) { return parent !== node && parent.contains(node) } : function(parent, node) { while (node && (node = node.parentNode)) if (node === parent) return true return false }
先来看看调用:
$.contains(parent, node)
参数
parent为父子点,
node为子节点。
$.contains的主体是一个三元表达式,返回的是一个匿名函数。三元表达式的条件是
document.documentElement.contains,
用来检测浏览器是否支持
contains方法,如果支持,则直接调用
contains方法,并且将
parent和
node为同一个元素的情况排除。
否则,返回另一外匿名函数。该函数会一直向上寻找
node元素的父元素,如果能找到跟
parent相等的父元素,则返回
true,
否则返回
false
$.grep
该函数其实就是数组的 filter函数
$.grep = function(elements, callback) { return filter.call(elements, callback) }
从源码中也可以看出,
$.grep调用的就是数组方法
filter
$.inArray
返回指定元素在数组中的索引值$.inArray = function(elem, array, i) { return emptyArray.indexOf.call(array, elem, i) }
先来看看调用
$.inArray(element, array, [fromIndex])
第一个参数
element为指定的元素,第二个参数为
array为数组,
第三个参数
fromIndex为可选参数,表示从哪个索引值开始向后查找。
$.inArray其实调用的是数组的
indexOf方法,所以传递的参数跟
indexOf方法一致。
$.isArray
判断是否为数组$.isArray = isArray
$.isArray调用的是内部方法
isArray,该方法在前一篇文章《读Zepto源码之内部方法》中已有阐述。
$.isFunction
判读是否为函数$.isFunction = isFunction
$.isFunction调用的是内部方法
isFunction,该方法在前一篇文章《读Zepto源码之内部方法》中已有阐述。
$.isNumeric
是否为数值$.isNumeric = function(val) { var num = Number(val), // 将参数转换为Number类型 type = typeof val return val != null && type != 'boolean' && (type != 'string' || val.length) && !isNaN(num) && isFinite(num) || false }
判断是否为数值,需要满足以下条件
不为
null
不为布尔值
不为NaN(当传进来的参数不为数值或如
'123'这样形式的字符串时,都会转换成NaN)
为有限数值
当传进来的参数为字符串的形式,如
'123'时,会用到下面这个条件来确保字符串为数字的形式,而不是如
123abc这样的形式。
(type != 'string' || val.length) && !isNaN(num)。这个条件的包含逻辑如下:如果为字符串类型,并且为字符串的长度大于零,并且转换成数组后的结果不为NaN,则断定为数值。(因为
Number('')的值为
0)
$.isPlainObject
是否为纯粹对象,即以 {}常量或
new Object()创建的对象
$.isPlainObject = isPlainObject
$.isPlainObject调用的是内部方法
isPlainObject,该方法在前一篇文章《读Zepto源码之内部方法》中已有阐述。
$.isWindow
是否为浏览器的 window对象
$.isWindow = isWindow
$.isWindow调用的是内部方法
isWindow,该方法在前一篇文章《读Zepto源码之内部方法》中已有阐述。
$.noop
空函数$.noop = function() {}
这个在需要传递回调函数作为参数,但是又不想在回调函数中做任何事情的时候会非常有用,这时,只需要传递一个空函数即可。
$.parseJSON
将标准JSON格式的字符串解释成JSONif (window.JSON) $.parseJSON = JSON.parse
其实就是调用原生的
JSON.parse,
并且在浏览器不支持的情况下,
zepto还不提供这个方法。
$.trim
删除字符串头尾的空格$.trim = function(str) { return str == null ? "" : String.prototype.trim.call(str) }
如果参数为
null或者
undefined,则直接返回空字符串,否则调用字符串原生的
trim方法去除头尾的空格。
$.type
类型检测$.type = type
$.type调用的是内部方法
type,该方法在前一篇文章《读Zepto源码之内部方法》中已有阐述。
相关文章推荐
- 读 zepto 源码之工具函数
- 读 zepto 源码之工具函数
- 读 zepto 源码之工具函数
- Zepto源码之$.fn工具函数
- Zepto源码之工具函数
- 读 zepto 源码之工具函数
- 读zepto源码之工具函数
- js函数格式化工具 及源码(转载)
- zepto 源码分析2 - 编码技巧 & 函数实现
- Vue源码阅读分享---工具函数
- jQuery源码分析(版本1.6.1)___构造jQuery对象-工具函数
- QUnit源码阅读(1):工具函数
- Zepto源码之辅助函数
- zepto源码研究 - zepto.js-3 (常用的工具)
- 【jQuery源码】工具函数
- 结合redis设计与实现的redis源码学习-26-工具函数(Util.h/.c)
- zepto源码学习-02 工具方法-详细解读
- jQuery源码学习(版本1.11)-事件处理-工具函数jQuery.event
- jquery源码分析-工具函数
- jquery技术揭秘静态工具函数源码重构