我的编程空间,编程开发者的网络收藏夹
学习永远不晚

比较JavaScript 对象的四种方式分别是怎样的

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

比较JavaScript 对象的四种方式分别是怎样的

比较JavaScript 对象的四种方式分别是怎样的,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。

 比较 JavaScript 中的值非常简单,只需用相等运算符即可,例如严格相等运算符:

'a' === 'c'; // => false 1   === 1;   // => true

但是对象却有结构化的数据,所以比较起来比较困难。在本文中,你将学习如何正确比较 JavaScript 中的对象。

1. 引用比较

JavaScript 提供了 3 种方法来对值进行比较:

  • 严格相等运算符 ===

  • 宽松相等运算符 ==

  • Object.is() 函数

当使用上述任何一种方法比较对象时,只有在比较的值引用了相同的对象实例时,比较的结果为 true。这就是参照相等性。

让我们定义对象 hero1 和 hero2 ,并查看实际中的参照相等性:

const hero1 = {    name: 'Batman'  };  const hero2 = {    name: 'Batman'  };    hero1 === hero1; // => true  hero1 === hero2; // => false  hero1 == hero1; // => true hero1 == hero2; // => false  Object.is(hero1, hero1); // => true Object.is(hero1, hero2); // => false

hero1 === hero1 的计算结果为 true,因为两个操作数均指向了同一个对象实例hero1。

另一方面,hero1 === hero2 的计算结果为 false,因为 hero1 和 hero2 是不同的对象实例。

有意思的是,hero1 和 hero2 对象的内容是相同的:两个对象都有一个 name 属性,它的其值是  'Batman'。尽管如此,即使比较相同结构的对象,hero1 === hero2 的结果也是 false。

当你想比较对象引用而不是它们的内容时,引用相等是很有用的。但是在更多的情况之下,你都想针对对象的实际内容进行比较:例如属性及它们的值。

接下来看看如何通过对象的内容比较对象是否相等。

2. 手动比较

按内容比较对象最直接的方法是读取属性并手动比较它们。

例如,让我们编写一个特殊的函数 isHeroEqual() 来比较两个 hero 对象:

function isHeroEqual(object1, object2) {    return object1.name === object2.name;  }    const hero1 = {    name: 'Batman'  };  const hero2 = {    name: 'Batman' }; const hero3 = {   name: 'Joker' };  isHeroEqual(hero1, hero2); // => true isHeroEqual(hero1, hero3); // => false

isHeroEqual() 访问两个对象的属性 name 并比较它们的值。

如果被比较的对象具有一些属性,我更喜欢编写诸如 isHeroEqual()  之类的比较函数。这类函数具有良好的性能:在比较中只会涉及少数几个属性访问器和相等运算符。

手动比较需要手动提取属性,对于简单对象来说,这不是问题。但是,要对较大的对象(或结构未知的对象)进行比较,就不方便了,因为它需要大量的样板代码。

那么让我们来看看对象的浅层比较能提供哪些帮助。

3. 浅层比较

如果用浅层比较检查对象,你必须获取两个对象的属性列表(使用Object.keys()),然后检查它们的属性值是否相等。

下面的代码是浅层比较的一种实现方式:

 function shallowEqual(object1, object2) {    const keys1 = Object.keys(object1);    const keys2 = Object.keys(object2);      if (keys1.length !== keys2.length) {      return false;    }      for (let index = 0; index < keys1.length; index++) {     const val1 = object1[keys1[index]];     const val2 = object2[keys2[index]];     if (val1 !== val2) {       return false;     }   }    return true; }

在函数内部,keys1 和 keys2 是分别包含 object1 和 object2 属性名称的数组。

用 for 循环遍历键,并比较 object1 和 object2 的每个属性。

使用浅层比较,你可以轻松对有着许多属性的对象进行相等性检查:

const hero1 = {   name: 'Batman',   realName: 'Bruce Wayne'  };  const hero2 = {    name: 'Batman',    realName: 'Bruce Wayne'  };  const hero3 = {   name: 'Joker' };  shallowEqual(hero1, hero2); // => true shallowEqual(hero1, hero3); // => false

shallowEqual(hero1, hero2) 返回 true,因为对象 hero1 和 hero2 具有相同的属性(name 和  realName),并且值也相同。

另一方面,由于 hero1 和 hero3 具有不同的属性,所以 shallowEqual(hero1, hero3) 将会返回 false。

但是 JavaScript 中的对象是可以嵌套的。在这种情况下,浅层比较并不能很好地发挥作用。

下面对具有嵌套对象的对象执行浅层比较检查:

 const hero1 = {   name: 'Batman',  address: {    city: 'Gotham'   } }; const hero2 = {  name: 'Batman',   address: {     city: 'Gotham'   } };  shallowEqual(hero1, hero2); // => false

这次,即使两个对象 hero1 和 hero2 具有相同的内容,shallowEqual(hero1, hero2) 也将会返回 false。

发生这种情况是因为嵌套对象 hero1.address 和 hero2.address 是不同的对象实例。因此,浅层比较认为 hero1.address  和 hero2.address 是两个不同的值。

解决嵌套对象的问题需要进行深层比较。

4. 深层比较

深层比较与浅层比较相似,不同之处在于,当属性中包含对象时,将对嵌套对象执行递归浅层比较。

看一下深层比较的实现:

function deepEqual(object1, object2) {   const keys1 = Object.keys(object1);  const keys2 = Object.keys(object2);     if (keys1.length !== keys2.length) {     return false;   }      for (let index = 0; index < keys1.length; index++) {     const val1 = object1[keys1[index]];     const val2 = object2[keys2[index]];     const areObjects = isObject(val1) && isObject(val2);     if (areObjects && !deepEqual(val1, val2) ||          !areObjects && val1 !== val2) {       return false;     }   }    return true; }  function isObject(object) {   return object != null && typeof object === 'object'; }

第 13 行的 areObjects && !deepEqual(val1, val2)  一旦检查到的属性是对象,则递归调用将会开始验证嵌套对象是否也相等。

现在用 deepEquality() 比较具有嵌套对象的对象:

const hero1 = {    name: 'Batman',    address: {      city: 'Gotham'   }  };  const hero2 = {   name: 'Batman',    address: {     city: 'Gotham'   } };  deepEqual(hero1, hero2); // => true

深度比较函数能够正确地确定 hero1 和 hero2 是否具有相同的属性和值,包括嵌套对象 hero1.address 和 hero2.address  的相等性。

为了深入比较对象,我建议使用Node内置util模块的 isDeepStrictEqual(object1, object2)(

https://nodejs.org/api/util.html#

util_util_isdeepstrictequal_val1_val2) 或lodash 库的_.isEqual(object1,  object2)(

https://lodash.com/docs/4.17.15#isEqual) 。


引用相等性(使用 ===、 == 或 Object.is())用来确定操作数是否为同一个对象实例。

手动检查对象是否相等,需要对属性值进行手动比较。尽管这类检查需要手动编码来对属性进行比较,但由于很简单,所以这种方法很方便。

当被比较的对象有很多属性或在运行时确定对象的结构时,更好的方法是使用浅层检查。

如果比较的对象具有嵌套对象,则应该进行深度比较检查。

看完上述内容是否对您有帮助呢?如果还想对相关知识有进一步的了解或阅读更多相关文章,请关注编程网行业资讯频道,感谢您对编程网的支持。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

比较JavaScript 对象的四种方式分别是怎样的

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

JavaScript的三种BOM对象分别是怎样的

这篇文章将为大家详细讲解有关JavaScript的三种BOM对象分别是怎样的,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。window 对象给我们提供了一个 location 属性用于获取或
2023-06-21

JavaScript对象的四种使用方式是什么

这篇“JavaScript对象的四种使用方式是什么”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“JavaScript对象的四
2023-06-30

Python bs4的四种对象分别是什么

小编给大家分享一下Python bs4的四种对象分别是什么,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!python可以做什么Python是一种编程语言,内置了许多有效的工具,Python几乎无所不能,该语言通俗易懂、容易
2023-06-14

JavaScript 对象创建的3种方法分别是什么

本篇文章给大家分享的是有关JavaScript 对象创建的3种方法分别是什么,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。前言:在 JavaScript中,对象是一组有属性名和
2023-06-25

横向对比分析Python解析XML的四种方式

在最初学习PYTHON的时候,只知道有DOM和SAX两种解析方法,但是其效率都不够理想,由于需要处理的文件数量太大,这两种方式耗时太高无法接受。 在网络搜索后发现,目前应用比较广泛,且效率相对较高的ElementTree也是一个比较多人推荐
2022-06-04

Kubernetes几种存储方式性能对比是怎样的

Kubernetes几种存储方式性能对比是怎样的,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。摘要展示了一个简单的存储对比,使用未经性能优化的多种存储提供的存储卷进行测试和比较
2023-06-04

Suse9故障修复的四种情景分别是是怎样的

Suse9故障修复的四种情景分别是是怎样的,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。随着Suse越来越受到广大用户的接受,伴随着问题也接踵而来,此次所讲的是
2023-06-17

Python中引用传参的四种方式分别是什么

这篇文章将为大家详细讲解有关Python中引用传参的四种方式分别是什么,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。引用传参一:>>> a = 100 #这里的a是不可变类型>>> def
2023-06-22

vue3组件通信的几种方式分别是这样的

vue3组件通信的几种方式分别是这样的,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。vue3组件通信方式为以下几种props$emit$expose / ref$attrsv-
2023-06-22

JavaScript中检测数据类型的四种方法分别是什么

这期内容当中小编将会给大家带来有关JavaScript中检测数据类型的四种方法分别是什么,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。前言:在介绍检测数据类型的方法之前,先说说JavaScript中数据类
2023-06-26

CentOS挂载NTFS分区的两种方法分别是怎样的

这篇文章给大家介绍CentOS挂载NTFS分区的两种方法分别是怎样的,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。第一种是安装内核模块,可到 http://sourceforge.net/projects/linux-
2023-06-10

编程热搜

目录