比较 JavaScript 中的数组对象

发布于 2022-09-02 19:16:43 字数 3568 浏览 144 评论 0

数组是 JavaScript 中的对象,所以 三等号运算符 === 只返回 true 如果数组是相同的引用。

const a = [1, 2, 3];
const b = [1, 2, 3];

a === a; // true
a === b; // false

如何比较两个数组是否相等? 相等是一个棘手的主题:JavaScript 规范定义了 4 种不同的方法来检查两个值是否 相等,并且没有考虑对象之间的深度相等。

在这种情况下,尽可能明确地说明 平等 的含义是有帮助的。 在软件工程中,以正确的方式提出问题通常会使答案显而易见。

考虑到这一点,这里有 3 个数组相等的定义以及如何检查它们。

长度相同,每个值相等

一种比较方法 ab 正在检查是否每个值 a 严格等于对应的值 b, 如果数组的所有元素都是基元而不是对象,则此方法效果很好。

const a = [1, 2, 3];
const b = [4, 5, 6];
const c = [1, 2, 3];

function arrayEquals(a, b) {
  return Array.isArray(a) &&
    Array.isArray(b) &&
    a.length === b.length &&
    a.every((val, index) => val === b[index]);
}

arrayEquals(a, b); // false
arrayEquals(a, c); // true

深度平等 POJO

以前的 arrayEquals() 函数适用于原始值,但如果您想按值比较对象,则效果不佳。

const a = [{ answer: 42 }, { powerLevel: 9001 }];
const b = [{ answer: 42 }, { powerLevel: 9001 }];

// false, because { answer: 42 } !== { answer: 42 }, different references
arrayEquals(a, b);

考虑对象值的一种巧妙方法是比较数组 JSON.stringify() 输出。

const a = [{ answer: 42 }, { powerLevel: 9001 }];
const b = [{ answer: 42 }, { powerLevel: 9001 }];
const c = [{ answer: 42 }, { password: 'taco' }];

JSON.stringify(a) === JSON.stringify(b); // true
JSON.stringify(a) === JSON.stringify(c); // false

这种方法很方便,因为它需要最少的代码并且不需要外部库。 然而比较 JSON.stringify() 输出有一个不幸的边缘情况,根据您的用例,这可能是一个问题。 自从 undefined 不是有效的 JSON 值,以下数组具有相同的 JSON.stringify() 输出,因为 JSON.stringify()转换 undefinednull

const a = [undefined];
const b = [null];

使用 Lodash 的 isEqual()

除了 null 对比 undefined 怪癖,比较 JSON.stringify() 输出也不考虑对象类型。 JSON.stringify() 是有关,一个对象与 toJSON() 返回 42 的函数与数字 42 相同。

const a = [{ toJSON: () => 42 }];
const b = [42];

JSON.stringify(a); // '[42]'
JSON.stringify(b); // '[42]'

同样,自定义对象与 POJO 相同:

class MyClass {
  constructor(obj) {
    Object.assign(this, obj);
  }
}

const a = [new MyClass({ answer: 42 })];
const b = [{ answer: 42 }];

JSON.stringify(a) === JSON.stringify(b); // true

Lodash 的 isEqual()function 另一方面:

const _ = require('lodash');

class MyClass {
  constructor(obj) {
    Object.assign(this, obj);
  }
}

const a = [new MyClass({ answer: 42 })];
const b = [{ answer: 42 }];

_.isEqual(a, b); // false

Lodash 的 isEqual() 如果您需要检查对象是否具有相同的类的所有花里胡哨的功能,那么函数就是您要走的路。JSON.stringify() 方法适用于 POJO,只要确保您考虑到 null 并且仅将其与受信任的数据一起使用 - toJSON() 可能是一个安全漏洞。

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

0 文章
0 评论
24 人气
更多

推荐作者

離殇

文章 0 评论 0

小姐丶请自重

文章 0 评论 0

Aik

文章 0 评论 0

国产ˉ祖宗

文章 0 评论 0

猥琐帝

文章 0 评论 0

半仙

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文