测试代码1:
var obj={ 0:"xx", 1:"female", length:2 } var arr=[1,2,3]; alert($.merge(arr,obj));//打印[1,2,3,xx,female] //不要求第二个参数是数组对象,但是obj前面的下标必须是0,1等数字,而且必须是从0开始的, //因为从源码就能看出来,j是从0开始的!这样才会出现j++不断加入如果obj前面不是数字, //或者数字不是从0开始的,那么获取到的数组就会[1,2,3,,]也就是有些属性不能添加进来!测试代码2:
var obj={ name:"xx", 1:"female", length:2 } var arr=[1,2,3]; alert($.merge(arr,obj)); //因为第二个对象的下标不是从0开始的!所以second[0]不存在,所以是空,最后打印[1,2,3,,female]测试代码3:
try { }catch(e) { alert(e.message);//用e.message输出错误信息! } //为了兼容IE<9,有些浏览器会把NodeLists元素的length转化为NAN!测试代码4:
var obj={'0':"zero","1":"one",length:2} alert(typeof obj.length);//返回number var obj={'0':"zero","1":"one",length:"2"} alert(typeof obj.length);//返回string,不要求length必须加上双引号
如果第二个参数的length可以转化为数字,那么 就转化为数字!
merge源码分析:
merge: function( first, second ) { //如果second.length是一个字符串,那么+second.length就是数字了,通过typeof +"123"将返回number //如果是+ "xx"那么就会返回0,typeof还是会返回number var len = +second.length, j = 0, i = first.length; //通过i不断的往里面添加,数组长度自动增加 //通过该循环,j就是第二个数组的长度 while ( j < len ) { first[ i++ ] = second[ j++ ]; } // Support: IE<9 // Workaround casting of .length to NaN on otherwise arraylike objects (e.g., NodeLists) //在IE<9中,会把NodeList等类数组对象的length转为NaN if ( len !== len ) { while ( second[j] !== undefined ) { first[ i++ ] = second[ j++ ]; } } //重新设置数组长度,数组的长度一般是number类型 first.length = i; return first; }
pushStack方法中用到了jQuery.merge方法:
pushStack: function( elems ) { // Build a new jQuery matched element set var ret = jQuery.merge( this.constructor(), elems );//this.constructor()=$()相当于返回一个空的jQuery对象,length是0,但是有很多属性方法! // Add the old object onto the stack (as a reference) ret.prevObject = this;//merge方法会获取到第二个参数的length,对于DOM对象来说第二个参数是undefined,所以什么也不做,因此把它 ret.context = this.context;//放在数组中才能真正用到merge方法!如$("span").pushStack([$("#n9")[0]])这样返回的jQuery对象就不是空的了! // Return the newly-formed element set return ret;//通过这里我们可以看到pushStack返回的是通过参数构建的jQuery对象,不过该jQuery对象保存了调用者context,以及通过prevObj } //保存了调用者的引用,该引用用于end方法!
总结:merge方法的逻辑还是比较简单的,但是要注意,merge方法的第二个参数可以不是数组(只要有length属性就可以了),甚至对象的键名可以不是数字,虽然这种情况将导致插入的全部是undefined!