当前位置: 首页 > ds >正文

JavaScript中判断两个对象是否相同(所有属性的值是否都相同)

在JavaScript中,判断两个对象的所有属性是否相同(包括嵌套对象)需要深度比较(deep comparison)。以下是几种实现方法:


方法1:简易深度比较(不考虑循环引用、Symbol和特殊对象)

javascript

复制

下载

function deepEqual(obj1, obj2) {// 基本类型直接比较if (obj1 === obj2) return true;// 检查是否为对象且非nullif (typeof obj1 !== 'object' || obj1 === null ||typeof obj2 !== 'object' || obj2 === null) {return false;}// 比较构造函数类型if (obj1.constructor !== obj2.constructor) return false;// 获取对象的键(仅自身可枚举属性)const keys1 = Object.keys(obj1);const keys2 = Object.keys(obj2);// 键数量不同则不等if (keys1.length !== keys2.length) return false;// 递归比较每个键的值for (const key of keys1) {if (!keys2.includes(key) || !deepEqual(obj1[key], obj2[key])) {return false;}}return true;
}

方法2:完整深度比较(支持循环引用、Symbol和特殊对象)

javascript

复制

下载

function deepEqual(obj1, obj2, visited = new WeakMap()) {// 基本类型直接比较if (obj1 === obj2) return true;// 检查是否为对象且非nullif (typeof obj1 !== 'object' || obj1 === null ||typeof obj2 !== 'object' || obj2 === null) {return false;}// 处理循环引用if (visited.has(obj1) {return visited.get(obj1) === obj2;}visited.set(obj1, obj2);// 比较构造函数类型if (obj1.constructor !== obj2.constructor) return false;// 特殊对象处理if (obj1 instanceof Date) return obj1.getTime() === obj2.getTime();if (obj1 instanceof RegExp) return obj1.toString() === obj2.toString();// 获取所有键(包括Symbol和自身可枚举属性)const keys1 = Reflect.ownKeys(obj1);const keys2 = Reflect.ownKeys(obj2);// 键数量不同则不等if (keys1.length !== keys2.length) return false;// 递归比较每个键的值for (const key of keys1) {if (!Reflect.has(obj2, key) ||!deepEqual(obj1[key], obj2[key], visited)) {return false;}}return true;
}

关键点说明

  1. 深度递归

    • 逐层比较嵌套对象的属性值。

    • 使用递归处理嵌套结构。

  2. 循环引用处理

    • 通过 WeakMap 记录已比较对象,避免无限递归。

    • 当检测到循环引用时直接比较引用是否相同。

  3. 特殊对象支持

    • Date:比较时间戳。

    • RegExp:比较正则表达式的字符串形式。

    • Array:自动支持(数组也是对象)。

  4. 键类型支持

    • 使用 Reflect.ownKeys() 获取所有自身键(包括Symbol)。

    • 使用 Object.keys() 仅获取可枚举字符串键(简易版)。

  5. 严格检查

    • 比较构造函数确保类型一致(如 Array vs Object)。

    • 键数量不等时快速失败。


使用示例

javascript

复制

下载

const objA = { a: 1, b: { c: 2 }, d: [3, 4] };
const objB = { a: 1, b: { c: 2 }, d: [3, 4] };
const objC = { a: 1, b: { c: 99 } };console.log(deepEqual(objA, objB)); // true
console.log(deepEqual(objA, objC)); // false

注意事项

  • 性能:深度比较可能较慢,避免在大型对象上使用。

  • 特殊对象:如需支持 Set/Map 等,需额外扩展。

  • 函数属性:函数按引用比较(通常不比较函数体)。

  • 库推荐:复杂场景建议使用 Lodash 的 _.isEqual()

javascript

复制

下载

// 使用 Lodash 的深度比较
import _ from 'lodash';
console.log(_.isEqual(objA, objB)); // true
http://www.xdnf.cn/news/12195.html

相关文章:

  • JavaWeb简介
  • Ansible常用模块和使用技巧
  • 学习笔记(23): 机器学习之数据预处理Pandas和转换成张量格式[1]
  • 前端css外边距塌陷(Margin Collapse)现象原因和解决方法
  • 【DAY39】图像数据与显存
  • Java 中创建线程主要有三种方式
  • Fast-dLLM:为扩散大模型按下加速键
  • 关于项目多语言化任务的概述
  • Manus AI 现在可以生成短片了
  • 电镀机的阳极是什么材质?
  • Windows系统下npm报错node-gyp configure got “gyp ERR“解决方法
  • 道可云人工智能每日资讯|人工智能赋能广西生态环境保护计划发布
  • JavaWeb:前端工程化-TS(TypeScript)
  • 鸿蒙任务项设置案例实战
  • 离散化思想
  • 链路聚合+VRRP
  • python入门(1)
  • 【.net core】【watercloud】树形组件combotree导入及调用
  • Visual Studio C++ 调试日志与异常定位指南
  • 时序替换实时?是否必要
  • 第16届蓝桥STEMA真题剖析-2025年4月13日Scratch初/中级组
  • OurBMC技术委员会2025年二季度例会顺利召开
  • Java创建多线程的四种方式
  • 使用osqp求解简单二次规划问题
  • 四元数:从理论基础到实际应用的深度探索
  • NNLM和word2vec的区别
  • 2025年文件加密软件推荐,最新款文档加密系统排名
  • (C++)STL:vector的认识与使用全解析
  • 70°视场+亚兆赫兹切换!硅光芯片上的「激光万花筒」登《Nature》封面
  • RDMA简介3之四种子协议对比