测试代码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!