zoukankan      html  css  js  c++  java
  • JavaScript如何比较两个数组的内容是否相同

    今天意外地发现JavaScript是不能用==或===操作符直接比较两个数组是否相等的。

    alert([]==[]);    // false
    alert([]===[]);   // false

    以上两句代码都会弹出false。

    因为JavaScript里面Array是对象,==或===操作符只能比较两个对象是否是同一个实例,也就是是否是同一个对象引用。目前JavaScript没有内置的操作符判断对象的内容是否相同。

    但是惯性思维让人以为数组也是值,是可以比较的。

    如果要比较数组是否相等,就只能遍历数组元素比较。

    在网上流传很普遍的一种做法是将数组转换成字符串:

    JSON.stringify(a1) == JSON.stringify(a2)

     或

    a1.toString() == a2.toString()

    请不要使用这种方法。

    这种方法在某些情况下是可行的,当两个数组的元素顺序相同且元素都可以转换成字符串的情况下确实可行,但是这样的代码存有隐患,比如数字被转换成字符串,数字“1”和字符串“1”会被认为相等,可能造成调试困难,不推荐使用。

    在StackOverflow上有大神已经提供了正确的方法,我就做下搬运工吧:

     1 // Warn if overriding existing method
     2 if(Array.prototype.equals)
     3     console.warn("Overriding existing Array.prototype.equals. Possible causes: New API defines the method, there's a framework conflict or you've got double inclusions in your code.");
     4 // attach the .equals method to Array's prototype to call it on any array
     5 Array.prototype.equals = function (array) {
     6     // if the other array is a falsy value, return
     7     if (!array)
     8         return false;
     9 
    10     // compare lengths - can save a lot of time 
    11     if (this.length != array.length)
    12         return false;
    13 
    14     for (var i = 0, l = this.length; i < l; i++) {
    15         // Check if we have nested arrays
    16         if (this[i] instanceof Array && array[i] instanceof Array) {
    17             // recurse into the nested arrays
    18             if (!this[i].equals(array[i]))
    19                 return false;       
    20         }           
    21         else if (this[i] != array[i]) { 
    22             // Warning - two different object instances will never be equal: {x:20} != {x:20}
    23             return false;   
    24         }           
    25     }       
    26     return true;
    27 }
    28 // Hide method from for-in loops
    29 Object.defineProperty(Array.prototype, "equals", {enumerable: false});

    大神还顺手给了比较Object的方法:

    Object.prototype.equals = function(object2) {
        //For the first loop, we only check for types
        for (propName in this) {
            //Check for inherited methods and properties - like .equals itself
            //https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/hasOwnProperty
            //Return false if the return value is different
            if (this.hasOwnProperty(propName) != object2.hasOwnProperty(propName)) {
                return false;
            }
            //Check instance type
            else if (typeof this[propName] != typeof object2[propName]) {
                //Different types => not equal
                return false;
            }
        }
        //Now a deeper check using other objects property names
        for(propName in object2) {
            //We must check instances anyway, there may be a property that only exists in object2
                //I wonder, if remembering the checked values from the first loop would be faster or not 
            if (this.hasOwnProperty(propName) != object2.hasOwnProperty(propName)) {
                return false;
            }
            else if (typeof this[propName] != typeof object2[propName]) {
                return false;
            }
            //If the property is inherited, do not check any more (it must be equa if both objects inherit it)
            if(!this.hasOwnProperty(propName))
              continue;
    
            //Now the detail check and recursion
    
            //This returns the script back to the array comparing
            /**REQUIRES Array.equals**/
            if (this[propName] instanceof Array && object2[propName] instanceof Array) {
                       // recurse into the nested arrays
               if (!this[propName].equals(object2[propName]))
                            return false;
            }
            else if (this[propName] instanceof Object && object2[propName] instanceof Object) {
                       // recurse into another objects
                       //console.log("Recursing to compare ", this[propName],"with",object2[propName], " both named ""+propName+""");
               if (!this[propName].equals(object2[propName]))
                            return false;
            }
            //Normal value comparison for strings and numbers
            else if(this[propName] != object2[propName]) {
               return false;
            }
        }
        //If everything passed, let's say YES
        return true;
    }  
  • 相关阅读:
    14.4 exportfs命令 14.5 NFS客户端问题 15.1 FTP介绍 15.2/15.3 使用vsftpd搭建ftp
    14.1 NFS介绍 14.2 NFS服务端安装配置 14.3 NFS配置选项
    13.4 mysql用户管理 13.5 常用sql语句 13.6 mysql数据库备份恢复
    13.1 设置更改root密码 13.2 连接mysql 13.3 mysql常用命令
    12.21 php-fpm的pool 12.22 php-fpm慢执行日志 12.23 open_basedir 12.24 php-fpm进程管理
    12.17 Nginx负载均衡 12.18 ssl原理 12.19 生成ssl密钥对 12.20 Nginx配置ssl
    12.13 Nginx防盗链 12.14 Nginx访问控制 12.15 Nginx解析php相关配置 12.16 Nginx代理
    在界面workspacebar上添加的OnContextMenu函数打的断点始终进不去,显示当前不会命中断点还未为文档加载任何符号
    VS2019安装闪退, 不出现安装界面,解决办法
    BCGP单元格多列合并需要注意的
  • 原文地址:https://www.cnblogs.com/-867259206/p/6795354.html
Copyright © 2011-2022 走看看