jQuery源码分析之$.map函数
来源:互联网 发布:凡科怎么绑定域名 编辑:程序博客网 时间:2024/05/19 02:21
<pre name="code" class="javascript">// 传入一个空函数作为参数,返回一个空数组// 空函数的返回值为undefined,而null或undefined值会被$.map()忽略掉。看最后的判断if(value!=null)var result = $.map( [1, 2, 3], $.noop );document.writeln( result.length ); // 0
测试代码1:
var a = {'1':'gg','2':'love','4':'meimei',length:5};//打印true,length是5alert(Array.prototype.slice.call(a) instanceof Array);//打印undefinedalert(Array.prototype.slice.call(a)[0]);测试代码2:
//callback可以多于两个参数var result=$.map([1,2],function(ele,index,input){// alert(input+"第三个参数的值:"+arguments[2]) if(ele<=2)return "返回值"+ele;},0)alert(result instanceof Array);//打印true,map返回Array类型。如果把上面if语句修改为if(elem<2)那么result只有一个结果"返回值1"alert(result[0]+result[1]);//打印"返回值1" "返回值2"测试代码3:
function Test(){alert("invoked!");}alert( Test());//返回值是undefined,也就是没有new对象的情况下,直接调用没有返回值的函数,返回undefined//$.map函数也只是返回符合条件的结果,否则没有返回值,那么调用callback就会返回undefined//alert(undefined!=null)//返回false,也就是当callback函数没有返回值的情况下,if(value!=null)是false,那么不会添加到数组中//否者value不是undefined,if条件满足了,直接添加进去
测试代码4:
//如果是对象该如何迭代呢?请看下面var obj={name:"xx",sex:"female"};//for(var i in obj){alert(i+"->"+obj[i])}//输出name->xx;sex->femalealert($.map(obj,function(elem,index,input){//alert(elem+"->"+elem[index]);//输出xx->undefined和female->undefined//如果是对象,那么就是拿着obj[i]来调用function这个函数,所以elem就是obj[i];//同时如下源码可以看出:第二个参数就是obj对象的属性名称,而不是属性值,所以index就是属性名称//for ( i in elems ) {//value = callback( elems[ i ], i, arg );// } },12))//如果是迭代数组呢?请看下面var arr=[1,2,3]$.map(arr,function(elem,index){alert(elem+"->"+index);//1->0,2->1,3->2。index就是下标,而elem就是arr[index].所以对于map函数来说第一个参数是值,第二个是键!})测试代码5:
$.map( [0,1,2], function(n){ return [ n, n + 1 ];});//输出:[0, 1, 1, 2, 2, 3]//源码如果是return ret的话,输出将会是:[[0,1], [1,2], [2,3]]//上面就是为什么源码要用concat的原因,因为他可以结合多个数组.(有元素才有下标所以元素在前,下标在后;同时map有return)var arr=[[0,1], [1,2], [2,3]];//打印6alert(Array.prototype.concat.apply([],arr).length);
$.map源码:
// arg is for internal usage onlymap: function( elems, callback, arg ) {var value,i = 0,length = elems.length,//必须是类数组,返回值为true或者false。拥有length属性,其它属性(索引)为非负整数(对象中的索引会被当做字符串来处理,//这里你可以当做是个非负整数串来理解)。不具有数组所具有的方法,这就是类数组! isArray = isArraylike( elems ),ret = [];// Go through the array, translating each of the items to their new values//如果是数组类型,从0开始遍历,对每一个对象单独调用callback函数,callback(elementOfArray,indexInArray)if ( isArray ) {for ( ; i < length; i++ ) {value = callback( elems[ i ], i, arg );if ( value != null ) {ret.push( value );}}// Go through every key on the object,} else {for ( i in elems ) {value = callback( elems[ i ], i, arg );//第一个是键值,第二个是键名!if ( value != null ) {ret.push( value );}}}return concat.apply( [], ret );}总结:
(1)在JS原生的map方法中也是有return语句的(forEach没有),原生的js中回调函数第一个参数是键值,第二个参数是键名,第三个参数是原生的数组。在jQuery中由于主要的代码为callback.call(elem[i],i,args)也就是说他的第一个参数也是键值,第二个参数也是键名,第三个参数是额外的参数(说明第三个参数有区别)。同时this也是有区别的,在JS的map中this指向map调用时候传入的第二个参数,如果传入为undefined或者null那么就是window,但是这里的this如果没有指定就是指向window对象!
在实例next,prev等方法的构建中有这样一句代码,对于理解jQuery.map方法特别有用:
var ret = jQuery.map( this, fn, until );其中的fn中用了三个参数来接受jQuery.map给该函数fn传入的参数,第一个是DOM,第二个是下标,第三个就是这里的until,表示同方向进行遍历的时候判断结束的元素
nextUntil: function( elem, i, until ) {return jQuery.dir( elem, "nextSibling", until );}
用这个额外的参数实现了如prevUntil,nextUntil,parentUntil实例函数的构建。同时还有一个问题,如果我们遍历调用对象的所有的parentNode,那么对于每一个调用对象的DOM都会形成一个DOM集合,那么返回的结果应该是[[elem],[elem],[elem]]类型,那么这部分的功能又是通过jQuery.map来完成的
var arr=[[1,2],[3,4]]; var result=[].concat.apply([],arr); //打印[1, 2, 3, 4] console.log(result);
(2)可以用Array.prototype.concat.apply([],arr)合并多个数组成为一个数组对象!
(3)map函数强调的是修改数组元素(只有返回值不是undefined才会放入结果数组中),grep强调的是筛选(和第三个参数比较,看返回值是true还是false),each强调的是遍历(用return false结束循环)
0 0
- jQuery源码分析之$.map函数
- jQuery源码分析之jQuery.makeArray函数
- 源码分析之jQuery.merge函数
- jQuery源码分析之proxy函数
- jQuery源码分析之$.inArray()函数
- jQuery源码分析之$.index函数
- jQuery源码分析之contains函数
- jQuery源码分析之globalEval函数
- jQuery源码之init函数的分析
- jQuery源码分析5: jQuery.map
- Map之HashMap源码分析
- Map之hashmap源码分析
- jquery源码分析之扩展函数 extend, $.extend
- jQuery源码分析之$.grep()函数四问
- jQuery源码分析之appendTo,prependTo,insertBefore,insertAfter,replaceWith函数
- jQuery源码分析之ajaxConverter与ajaxHandleResponse函数
- jQuery源码分析-each函数
- jQuery源码分析-extend函数
- 【Ray Trace from Groud Up】光线追踪代码实现解析
- 1002. A+B for Polynomials (25)
- 【笔试】53、圆圈中最后剩下的数字(约瑟夫环)
- 关于ios8中CMStepCounter过时的问题
- 纪念碑
- jQuery源码分析之$.map函数
- 用快排思路解决逆序对问题
- 基于TCP网络通信的简易多线程GUI聊天室
- ubuntu Linux离线安装软件包
- 斐波那契的递归函数
- JSP要点总结
- IOS开发UI进阶之UITableView三
- 最高分是多少(线段树) 华为2016校园招聘
- java并发之ConcurrentHashMap