jQuery静态方法inArray,grep,merge,makeArray方法使用和源码分析
inArray方法
确定第一个参数在数组中的位置,从0开始计数(如果没有找到则返回 -1 )。
示例:
var arr = [ 4, "Pete", 8, "John" ];
jQuery.inArray("John", arr); //
jQuery.inArray(4, arr); //
jQuery.inArray("David", arr); //-1
jQuery.inArray("Pete", arr, 2); //-1
源码分析:
inArray: function( elem, array, i ) {
var len;
if ( array ) {
if ( indexOf ) {
return indexOf.call( array, elem, i );
}
len = array.length;
i = i ? i < 0 ? Math.max( 0, len + i ) : i : 0;
for ( ; i < len; i++ ) {
// Skip accessing in sparse arrays
if ( i in array && array[ i ] === elem ) {
return i;
}
}
}
return -1;
},
接受3个参数,elem参数为查找的元素,array是待查找的数组i表示指定开始查找的位置,默认是 0 即查找整个数组 类似trim方法,ECMA5也给数组对象提供了一个原型方法indexOf,如果执行代码的浏览器支持此方法直接调用;
var arr=[1,2,3]; alert(arr.indexOf(2)); // alert(arr.indexOf(2,2)); //-1
对于不需要考虑低版本浏览的开发人员可以直接使用,对于不支持的浏览器使用for...in循环过滤;
首先把数组长度保存下来,根据长度对i进行处理,如果i不存在让其为0如果小于0加上len再判断,如果还是小于0则再次使其为0;
这里的for...in循环跟我们平常使用的形式不太一样,正如注释中所言要规避稀疏数组,什么是稀疏数组举例如下:
var arr=[undefined,undefined];
arr[2]=4;
console.log(arr);//[undefined,undefined,4]
var i=0;len=arr.length;
for ( i in arr ) {
alert(i in arr); //true x 3
} var arr=[];
arr[2]=4;
console.log(arr);//[2:4]
var i=0;len=arr.length;
for ( i in arr ) {
alert(i in arr); //false x 3 true
}
两个数组采用一样的方法,虽然长度都为3,表现值也一样,但是在使用in操作返回的结果是不一样的,第二种就是稀疏数组,其元素是不连续的,for...in循环是会自动添加两个undfined 但是结果却为false,对于这种类型的数组是不需要js后台添加的undefined的,所以jQuery采用了这种方式处理,这种方式也值得借鉴。找到之后返回对应i值否则返回-1,所以可以判断返回结果>-1进行判断是否存在。
grep方法
使用过滤函数过滤数组元素。
此函数至少传递两个参数:待过滤数组和过滤函数。过滤函数必须返回 true 以保留元素或 false 以删除元素。
示例:
$.grep( [0,1,2], function(n,i){
return n > 0;
});
//结果:
[1, 2]
//排除数组中大于 0 的元素,使用第三个参数进行排除。
$.grep( [0,1,2], function(n,i){
return n > 0;
}, true);
//结果:
[0]
源码分析:
grep: function( elems, callback, inv ) {
var ret = [], retVal;
inv = !!inv;
// Go through the array, only saving the items
// that pass the validator function
for ( var i = 0, length = elems.length; i < length; i++ ) {
retVal = !!callback( elems[ i ], i );
if ( inv !== retVal ) {
ret.push( elems[ i ] );
}
}
return ret;
}
该方法接受3个参数,elems为待过滤数组,callback是回调函数,inv表示是否反向过滤默认是false;
首先对inv强制转为布尔值,然后使用for循环,把数组里面的元素逐个传入回调函数中运行,返回结果赋值给retVal,如果retVal恒等于inv,就把该元素放到新数组ret中,最后返回ret,需要注意的是因为是恒等所以回调函数返回值一定要是布尔值。
merge方法
合并两个数组
返回的结果会修改第一个数组的内容——第一个数组的元素后面跟着第二个数组的元素。
示例:
//合并两个数组到第一个数组上。
$.merge( [0,1,2], [2,3,4] ) //结果:
[0,1,2,2,3,4]
源码分析:
merge: function( first, second ) {
var i = first.length,
j = 0;
if ( typeof second.length === "number" ) {
for ( var l = second.length; j < l; j++ ) {
first[ i++ ] = second[ j ];
}
} else {
while ( second[j] !== undefined ) {
first[ i++ ] = second[ j++ ];
}
}
first.length = i;
return first;
},
该方法结接受两个数组参数,第二个是作为合并数组,第一个是待合并数组。其实这里的数组并不一定是"纯数组",有可能是类数组或者带有数字下标的对象等;
首先根据第二个数组的length属性进行区分,如果是number类型则假定是数组,用for循环逐个添加到第一个数组上面去。如果不是number或者不存在,则采用while循环,把第二个数组中所又非undefined的值添加或者覆盖到第一个数组对应下下标值。
最后手动修正length值,因为对于非“纯数组”的数据而言,length值时不会自动修改的。
makeArray方法
将类数组对象转换为数组对象。
类数组对象有 length 属性,其成员索引为 0 至 length - 1。实际中此函数在 jQuery 中将自动使用而无需特意转换。
示例:
//HTML 代码:
<div>First</div><div>Second</div><div>Third</div><div>Fourth</div> //jQuery 代码:
var arr = jQuery.makeArray(document.getElementsByTagName("div"));
arr.reverse(); // 使用数组翻转函数 //结果:
[Fourth,Third,Second,First]
源码分析:
// results is for internal usage only
makeArray: function( array, results ) {
var ret = results || []; if ( array != null ) {
// The window, strings (and functions) also have 'length'
// Tweaked logic slightly to handle Blackberry 4.7 RegExp issues #6930
var type = jQuery.type( array ); if ( array.length == null || type === "string" || type === "function" || type === "regexp" || jQuery.isWindow( array ) ) {
push.call( ret, array );
} else {
jQuery.merge( ret, array );
}
} return ret;
},
这里接受两个参数,其中第二个参数是内部使用的,在源码内部经常被调用作为其他方法的支持方法。
创建一个新数组ret,如果只有一个参数则为空,如果存在第二个参数就把第二个参数赋值给ret,在array参数存在的前提下获取其数据类型,如果数据类型为字符串、函数或者正则时或者不存在length属性时,则假定array不是数组或者类数组,因为字符串、函数和正则(黑莓系统下)都是有length属性的,所以只判断length不准确,如果不是数组或者类数组就直接将第一个参数放入ret的末尾。如果通过了则认为是数组或者是类数组,此时调用merge方法将两个数组合并,最后返回ret。
jQuery静态方法inArray,grep,merge,makeArray方法使用和源码分析的更多相关文章
- jQuery静态方法isPlainObject,isEmptyObject方法使用和源码分析
isPlainObject方法 测试对象是否是纯粹的对象(通过 "{}" 或者 "new Object" 创建的) 示例: //测试是否为纯粹的对象 jQuer ...
- jQuery静态方法parseJSON方法使用和源码分析
该方法接受一个JSON字符串,返回解析后的对象. 传入一个畸形的JSON字符串会抛出一个异常.比如下面的都是畸形的JSON字符串: {test: 1} ( test 没有包围双引号) {'test': ...
- jQuery原型方法.pushStack源码分析
这次分析的方法跟前面不同,虽然pushStack也是原型方法之一,但是我们几乎从不用在页面调用,在参考手册里面也没有这个方法的使用说明,但是这个方法还是非常重要的,在使用很多jQuery的其他方式都会 ...
- jQuery静态方法noop,camelCase,nodeName,trim使用和源码分析
noop方法 jQuery.noop()函数是一个空函数,它什么也不做. 当某些时候你需要传入函数参数,而且希望它什么也不做的时候,你可以使用该函数,也无需再新建一个空的函数. 示例: // 传入一个 ...
- jQuery静态方法globalEval使用和源码分析
Eval函数大家都很熟悉,但是globalEval方法却很少使用,大多数参考手册也没有相关api,下面就对其用法和源码相应介绍: jQuery.globalEval()函数用于全局性地执行一段Java ...
- jQuery静态方法parseXML使用和源码分析
jQuery.parseXML( data ) 接受一个格式良好的 XML 字符串,返回解析后的 XML 文档. 方法 jQuery.parseXML() 使用浏览器原生的 XML 解析函数实现. 在 ...
- jQuery静态方法isFunction,isArray,isWindow,isNumeric使用和源码分析
上一篇随笔中总结了js数据类型检测的几个方法和jQuery的工具方法type方法,本篇要分析几个方法都依赖type方法,所以不了解type方法的请先参看http://www.cnblogs.com/y ...
- RxJava2 中多种取消订阅 dispose 的方法梳理( 源码分析 )
Github 相关代码: Github地址 一直感觉 RxJava2 的取消订阅有点混乱, 这样也能取消, 那样也能取消, 没能系统起来的感觉就像掉进了盘丝洞, 迷乱… 下面说说这几种情况 几种取消的 ...
- 【Android笔记】Thread类中关于join()方法的源码分析
1.join()方法的作用: 例如有一个线程对象为Thread1,在main()方法中调用Thread1.join()方法可使得当前线程(即主线程)阻塞,而执行Thread1线程. 2.源码分析(以上 ...
随机推荐
- Eclipse开发工具组合键介绍
1. 编辑类 Ctrl + Q 跳到最后一次的编辑处 Ctrl + 1 快速修复提示 Alt + ↓ ...
- c++与java中子类中调用父类成员的方法
java中: import java.util.Scanner; public class ClassTest{ public static void main(String args[]){ chi ...
- 如何用Node编写命令行工具
0. 命令行工具 当全局安装模块之后,我们可以在控制台下执行指定的命令来运行操作,如果npm一样.我把这样的模块称之为命令行工具模块(如理解有偏颇,欢迎指正) 1.用Node编写命令行工具 在Node ...
- EntityFramework之原始查询如何查询未映射的值,你又知道多少?
前言 今天又倒腾了下EF,如题所示,遇到了一些问题,并最终通过尝试找到了解决方案,可能不是最终的解决方案,若你有更好的解决方案,望告知,同时通过阅读此文,定让你收获不少. 引入 当我们查询时一直是中规 ...
- iframe跨域+
script.image.iframe的src都不受同源策略的影响.所以我们可以借助这一特点,实现跨域.如前面所介绍的JSONP跨域,以及灯标(Beacons). 该篇随笔主要阐述iframe结合一些 ...
- iOS开发之地图与定位
无论是QQ还是微信的移动客户端都少不了定位功能,之前在微信demo中没有添加定位功能,今天就写个定位的小demo来了解一下定位和地图的东西.地图和定位看上去是挺高大上一东西,其实用法比TableVie ...
- 【jQuery小实例】---3 凤凰网首页图片动态效果
---本系列文章所用使用js均可在本博客文件中找到 本页面实现类似于凤凰网首页,鼠标点击新闻,可以在div中显示新闻图片,点击军事显示军事图片的效果.采用的思路是:鼠标悬浮,显示当前div中的内容(图 ...
- 使用Oracle调度程序自动完成任务
1. 创建作业.计划和时间表 2. 创建轻量级作业 3. 使用作业链执行一系列相关任务 4. 创建窗口和作业类 5. 使用高级调度程序概念确定作业优先顺序 Reference 实验演示准备: --业务 ...
- 1Z0-053 争议题目解析175
1Z0-053 争议题目解析175 考试科目:1Z0-053 题库版本:V13.02 题库中原题为: 175.You are peer reviewing a fellow DBAs backup p ...
- Node.js入门初体验
今天有一个类似网络爬虫的需求,本来打算用我还算熟悉的asp或者asp.NET来做这个事情,但是写了这么长时间js,asp的语法实在不喜欢,VS又早被我卸掉了,思来想去打算用一下最近比较火的Node.j ...