zoukankan      html  css  js  c++  java
  • jQuery静态方法inArray,grep,merge,makeArray方法使用和源码分析

    inArray方法

     

    确定第一个参数在数组中的位置,从0开始计数(如果没有找到则返回 -1 )。

    示例:

    var arr = [ 4, "Pete", 8, "John" ];
    
    jQuery.inArray("John", arr);  //3
    
    jQuery.inArray(4, arr);  //0
    
    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));  //1
         
      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。

  • 相关阅读:
    安装CentOS7重启后提示License information
    使用VMware 安装Linux CentOS7
    VS 2015相当不错的功能:C#交互窗口
    未能正确加载“Microsoft.VisualStudio.Editor.Implementation.EditorPackage”包
    RabbitMQ service is already present
    RabbitMQ安装后不能运行 Error: unable to connect to node nodedown
    〖Demo〗-- 计算器
    〖Demo〗-- HAproxy配置文件操作
    〖Python〗-- 模块系列(二)
    〖Python〗-- 模块系列(一)
  • 原文地址:https://www.cnblogs.com/yy-hh/p/4680168.html
Copyright © 2011-2022 走看看