js 怎样合并两个对象

在javascript中合并对象最推荐的方式是使用展开语法或object.assign()方法,1. 展开语法通过{…obj1, …obj2}创建新对象,不修改原对象,符合不可变性原则;2. object.assign()通过object.assign(target, source1, source2)将源对象属性复制到目标对象,若目标为空则实现合并,否则会修改目标对象;3. 两者均执行浅拷贝,嵌套对象仅复制引用,需手动递归或使用lodash的merge实现深拷贝;4. 属性冲突时遵循“后覆盖前”规则,右侧对象属性优先;5. 合并多个对象时,两种方法均支持链式或参数列表方式,按顺序从左到右合并,后出现的属性值覆盖前面同名属性,最终返回合并后的对象。

js 怎样合并两个对象

在JavaScript里合并两个对象,最常见也是最推荐的方式,通常是利用展开语法(spread syntax)或者

Object.assign()

方法。这两种方式各有侧重,但都能高效地将多个对象的属性汇集到一起,形成一个新的对象,或者更新一个现有对象。

解决方案

说起JavaScript里对象合并,这事儿看似简单,但真要抠细节,还是有些门道的。我们最常用的,也是我个人觉得在现代JS开发中最舒服的方式,就是展开语法(spread syntax)

想象一下,你有一堆零散的积木(对象的属性),你想把它们都放到一个新的盒子里。展开语法就是那个能帮你把积木从旧盒子里“倒”出来,再“铺”到新盒子里的工具

const obj1 = { a: 1, b: 2 };const obj2 = { c: 3, d: 4 };// 使用展开语法合并const mergedObj = { ...obj1, ...obj2 };console.log(mergedObj); // { a: 1, b: 2, c: 3, d: 4 }

这种方式的优点在于它创建了一个全新的对象,原始对象不会被修改,这符合函数式编程中“不可变性”的理念,让代码更可预测,尤其是在状态管理中,这简直是福音。它就像是制作一份新的文件副本,而不是直接在原文件上涂改。

当然,还有个老牌选手,

Object.assign()

方法。它就像是一个工厂的工人,负责把源对象的属性复制到目标对象上。

const objA = { x: 10, y: 20 };const objB = { z: 30 };// 使用Object.assign()合并const targetObj = {}; // 通常我们会提供一个空对象作为目标Object.assign(targetObj, objA, objB);console.log(targetObj); // { x: 10, y: 20, z: 30 }// 也可以直接合并到现有对象上,但会修改它const existingObj = { name: 'Alice' };Object.assign(existingObj, { age: 30 });console.log(existingObj); // { name: 'Alice', age: 30 }

Object.assign()

的第一个参数是目标对象,后面的参数是源对象。它会将所有源对象的可枚举属性复制到目标对象上,并返回目标对象。要注意的是,如果第一个参数不是一个空对象,那么这个对象本身会被修改。在某些场景下,你可能就想原地修改一个对象,那

Object.assign()

就很合适。但如果你追求不可变性,记得传入一个空对象作为第一个参数。

这两种方法,无论是展开语法还是

Object.assign()

,它们都只执行浅拷贝。这意味着如果你的对象里嵌套了其他对象或数组,合并时只会复制它们的引用,而不是递归地复制它们的内容。这一点,我们待会儿可以展开聊聊,因为这常常是初学者踩坑的地方。

合并对象时,属性冲突如何解决?

这是个很实际的问题,毕竟不是每次合并都那么风平浪静,属性名一样的情况太常见了。当两个或多个对象在合并时拥有相同的属性名,JavaScript有一套清晰的规则来处理这种“冲突”,简单来说就是“后来者居上”。

无论是使用展开语法还是

Object.assign()

,它们的行为都是一致的:位于后面(或右侧)的对象的同名属性值,会覆盖前面(或左侧)对象的属性值。

我们来看个例子:

const userDefault = {  name: 'Guest',  role: 'viewer',  permissions: ['read']};const userConfig = {  name: 'John Doe',  permissions: ['write', 'delete'],  status: 'active'};// 使用展开语法const finalUserSpread = { ...userDefault, ...userConfig };console.log(finalUserSpread);/*{  name: 'John Doe',     // userConfig 的 name 覆盖了 userDefault 的  role: 'viewer',  permissions: ['write', 'delete'], // userConfig 的 permissions 覆盖了 userDefault 的  status: 'active'}*/// 使用Object.assign()const finalUserAssign = Object.assign({}, userDefault, userConfig);console.log(finalUserAssign);/*{  name: 'John Doe',     // userConfig 的 name 覆盖了 userDefault 的  role: 'viewer',  permissions: ['write', 'delete'], // userConfig 的 permissions 覆盖了 userDefault 的  status: 'active'}*/

从这个例子可以看到,

name

permissions

属性都被

userConfig

中的值覆盖了。这很符合直觉,也常用于配置合并的场景:先定义一个默认配置,然后用用户提供的配置去覆盖或补充它。

这种“覆盖”机制在很多情况下都非常有用,比如当你需要合并一个基础配置对象和用户自定义的特定配置时。你只需要把默认值放在前面,把个性化设置放在后面,就能轻松实现配置的优先级管理。但同时也要注意,如果你希望合并的属性是某种“累加”而不是“覆盖”,比如两个数组属性,那这种默认的合并行为就帮不上忙了,你需要自己写额外的逻辑来处理。这正是“浅拷贝”特性带来的影响,我们接着聊。

深拷贝与浅拷贝:合并嵌套对象时需要注意什么?

前面提到,无论是展开语法

...

还是

Object.assign()

,它们都执行的是浅拷贝。这意味着什么呢?简单来说,它们只会复制对象的第一层属性。如果属性值是基本类型(字符串、数字、布尔值、null、undefined、Symbol、BigInt),那它们会被直接复制一份。但如果属性值是引用类型(对象、数组、函数),那么复制的将是这个引用本身的地址,而不是引用指向的实际内容。

这听起来有点抽象,我们直接看个例子来感受一下这个“坑”:

const userProfile = {  id: 1,  name: 'Jane',  address: {    street: '123 Main St',    city: 'Anytown'  },  hobbies: ['reading', 'hiking']};const updates = {  name: 'Jane Doe',  address: {    city: 'Newville' // 假设我们只更新城市  },  hobbies: ['coding'] // 假设我们想添加一个爱好};const mergedProfile = { ...userProfile, ...updates };console.log(mergedProfile);/*{  id: 1,  name: 'Jane Doe',  address: { city: 'Newville' }, // 注意:street 属性不见了!  hobbies: ['coding'] // 注意:hiking 爱好不见了!}*/// 更糟糕的是,如果更新的是引用类型内部的属性const userProfile2 = {  id: 2,  name: 'Bob',  settings: {    theme: 'dark',    notifications: {      email: true,      sms: false    }  }};const updates2 = {  settings: {    notifications: {      sms: true // 试图只更新sms通知    }  }};const mergedProfile2 = { ...userProfile2, ...updates2 };console.log(mergedProfile2);/*{  id: 2,  name: 'Bob',  settings: {    notifications: { sms: true } // 注意:theme 属性和 email 通知都不见了!  }}*/

看到了吗?当

updates

对象中的

address

属性是一个新的对象时,它会完全替换掉

userProfile

中的

address

对象,而不是合并它们内部的属性。同样,

hobbies

数组也被整个替换了。在第二个例子里,

settings

notifications

也是如此。这种行为可能不是你想要的,你可能希望的是深度合并,即递归地合并嵌套对象和数组的属性。

要实现深拷贝合并,

...

Object.assign()

就无能为力了。你通常需要:

手动递归合并: 编写一个函数,遍历对象的每一个属性。如果属性值是对象或数组,就递归调用自身进行合并。这需要一些代码量,但能完全控制合并逻辑。使用第三方库: 像Lodash这样的实用工具库提供了

_.merge()

_.mergeWith()

这样的方法,它们能够方便地进行深度合并,并且通常考虑了各种边缘情况。在实际项目中,这往往是更省心且健壮的选择。

所以,当你处理包含嵌套对象或数组的数据时,一定要先问自己:我需要的是浅拷贝还是深拷贝?如果答案是深拷贝,那么请记住,原生的

...

Object.assign()

并不是你的终极解决方案,它们只是起点。

合并多个对象,除了两个,还有哪些高效方法?

合并两个对象是基础,但在实际应用中,我们经常需要合并三个、四个甚至更多个对象。好消息是,我们前面提到的两种主要方法——展开语法和

Object.assign()

——都非常擅长处理这种情况,而且用法几乎一样直观。

先看展开语法

const baseConfig = {  port: 3000,  env: 'development',  logging: true};const devConfig = {  env: 'development',  debug: true,  port: 8080 // 覆盖 baseConfig 的 port};const userOverrides = {  logging: false, // 覆盖 baseConfig 的 logging  debug: false // 覆盖 devConfig 的 debug};// 将所有对象按顺序展开,后面的对象会覆盖前面同名属性const finalConfig = { ...baseConfig, ...devConfig, ...userOverrides };console.log(finalConfig);/*{  port: 8080,      // 被 devConfig 覆盖  env: 'development',  logging: false,  // 被 userOverrides 覆盖  debug: false     // 被 userOverrides 覆盖}*/

这种链式的展开方式非常清晰,一眼就能看出合并的顺序和优先级。从左到右,后面的对象属性会覆盖前面的同名属性,这在处理多层配置或选项时非常方便。

接着是

Object.assign()

const baseSettings = { theme: 'light', font: 'sans-serif' };const userSettings = { theme: 'dark' };const adminSettings = { permissions: ['all'], font: 'monospace' };// Object.assign(target, source1, source2, ..., sourceN)// 第一个参数是目标对象,后续所有参数都是源对象const combinedSettings = Object.assign({}, baseSettings, userSettings, adminSettings);console.log(combinedSettings);/*{  theme: 'dark',        // 被 userSettings 覆盖  font: 'monospace',    // 被 adminSettings 覆盖  permissions: ['all']}*/

Object.assign()

同样支持传入任意数量的源对象。它会按照参数的顺序,依次将每个源对象的可枚举属性复制到目标对象上。因此,越靠后的源对象,其属性的优先级越高。

从效率上讲,对于大多数日常使用场景,这两种方法在合并少量到中等数量的对象时,性能差异微乎其微,你完全可以根据个人偏好和团队规范来选择。展开语法通常被认为是更现代、更简洁的写法,尤其是在需要创建新对象时。而

Object.assign()

在需要原地修改现有对象时,或者在旧版浏览器兼容性要求较高(虽然现在展开语法也基本普及了)的情况下,依然是可靠的选择。

总结一下,无论是合并两个还是多个对象,JavaScript都提供了简洁且高效的原生方法。关键在于理解它们的浅拷贝特性以及属性冲突的解决机制,这样才能在实际开发中避免不必要的“惊喜”。

以上就是js 怎样合并两个对象的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/96128.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月19日 20:38:04
下一篇 2025年11月19日 21:05:36

相关推荐

  • PHP中CI框架的运行模式

    CodeIgniter的“运行模式”并非严格定义,而是通过配置和代码控制的执行流程风格:1. 单入口模式为默认方式,所有请求经index.php进入,由CodeIgniter.php驱动完整生命周期;2. CLI模式支持命令行执行,用于定时任务等场景,通过is_cli()识别环境并跳过Web专属逻辑…

    2025年12月13日
    000
  • php源码怎么进后台_php源码进入后台与权限设置【方法】

    首先确认后台入口路径,检查常见目录如admin或登录文件;接着通过数据库查看或修改管理员账号密码;再检查配置文件开启调试模式或添加会话绕过登录;然后设置正确文件权限确保可读写;最后尝试URL参数触发隐藏激活机制。 如果您拥有PHP源码项目并希望进入其后台管理系统,但不清楚如何访问或配置权限,则可能是…

    2025年12月13日
    000
  • php怎么通过md5解密出来_用PHP逆向md5加密获取明文教程【技巧】

    MD5不可逆,无法直接还原,但可通过彩虹表查询、本地字典比对或在线API批量查询尝试匹配明文。 如果您尝试对一个经过MD5处理的字符串进行还原,需要明确的是MD5是一种单向哈希算法,设计目的就是不可逆。因此无法通过传统意义上的“解密”来直接还原原始数据。但可以通过查找已知明文与对应哈希值的方式尝试匹…

    2025年12月13日
    000
  • php源码怎么安装_用PHP环境安装源码步骤教程【教程】

    首先检查并搭建PHP运行环境,安装XAMPP等集成环境,启动Apache和MySQL服务,将源码放入htdocs或www目录,通过localhost访问项目;接着配置数据库连接信息,修改config.php等文件中的数据库参数,创建数据库并导入SQL文件;然后设置文件权限,确保uploads、cac…

    2025年12月13日
    000
  • 找php源码怎么破解_找php源码破解风险与合法建议【警示】

    使用PHP源码破解工具违法且危险,一、非法破解侵犯知识产权并可能植入后门,需检查来源、分析可疑函数、扫描病毒;二、避免第三方解密工具,拒传源码、禁用未知脚本、排查新文件;三、应通过开源平台、购买授权或自研代码合法获取;四、加强防护,关闭危险函数、限制目录权限、启用Open_basedir。 如果您在…

    2025年12月13日
    000
  • ZKTeco考勤数据集成至Google Sheets或在线服务器实战教程

    本教程详细介绍了如何将ZKTeco考勤机(如K40、F18)的考勤数据集成到Google Sheets或自定义在线服务器。通过开发中间程序获取设备数据,并在服务器端进行处理,最终利用Google Apps Script的UrlFetch服务将JSON数据导入Google Sheets,有效解决了考勤…

    2025年12月13日
    000
  • 解决CodeIgniter 4中表单提交后重定向失败的CSP问题

    当CodeIgniter 4应用中的表单提交后重定向功能失效,尤其是在Chrome浏览器中出现“Refused to send form data… form-action ‘self’”的错误时,这通常是由于内容安全策略(CSP)中的`form-action &…

    2025年12月13日
    000
  • 在PHP中安全处理并发送多选框(Checkbox)值到邮件

    本教程详细阐述了如何在PHP中正确处理HTML表单提交的多选框(Checkbox)数组值,并将其格式化以便通过邮件发送。文章涵盖了从HTML结构到PHP数组处理的关键步骤,包括使用`implode`函数将数组转换为字符串,以及至关重要的安全实践,如使用`htmlentities`或`htmlspec…

    2025年12月13日
    000
  • 使用Ajax从超链接动态传递GET参数到PHP页面

    本教程详细讲解如何利用jquery ajax技术,从html超链接的`href`属性中动态获取get参数,并将其发送至php后端进行处理,从而实现页面无刷新数据交互。文章将涵盖从前端javascript拦截链接点击、构建ajax请求到后端php接收数据的完整流程,并提供示例代码和注意事项。 在Web…

    2025年12月13日 好文分享
    000
  • 使用 Guzzle HTTP 和 Goutte 模拟表单登录教程

    直接使用 Guzzle 的 `auth` 选项通常不适用于模拟基于表单的网站登录。本教程将指导您如何通过模拟浏览器行为实现表单登录,包括首先通过 GET 请求获取登录页面以提取表单数据(如 CSRF 令牌),然后使用 POST 请求提交凭据,并确保会话管理以进行后续的认证请求。 在开发涉及与外部网站…

    2025年12月13日
    000
  • SQL与PHP实现课程学生并发量精确统计教程

    本教程详细阐述了在mysql 5.6和php 7.2环境下,如何精确统计指定课程在特定日期范围内的学生并发量。针对传统查询无法准确处理日期区间重叠的问题,文章提出并演示了利用“日历表”结合sql聚合函数,有效计算每日活跃学生数,并从中找出指定时间段内的最大并发峰值,确保统计结果的准确性与可靠性。 挑…

    2025年12月13日
    000
  • PHP密码长度验证:常见陷阱与最佳实践

    本教程详细探讨php中密码长度验证的正确实现方法,重点纠正常见的逻辑错误,并强调使用`mb_strlen`处理多字节字符的重要性。文章将指导开发者如何构建健壮且易读的验证函数,并将其无缝集成到表单提交流程中,同时提供代码示例和优化建议,确保密码验证的准确性和安全性。 密码长度验证的重要性与常见误区 …

    2025年12月13日
    000
  • Nginx自定义错误页面:实现外部跳转与邮件通知

    本文详细阐述如何在nginx中配置自定义错误页面,以实现当服务器出现4xx或5xx错误时,不仅能将用户重定向到指定的外部网站,还能同时触发后端php脚本发送邮件通知。教程涵盖nginx配置、php脚本实现及curl命令行测试方法,旨在提供一个灵活且功能强大的错误处理机制。 在网站运维中,优雅地处理服…

    2025年12月13日
    000
  • 利用PHP DateTime类处理日期输入与月份识别

    本文旨在解决%ignore_a_1%中根据用户提交日期判断其所属月份的问题。传统的多条件if-else或switch-case语句在处理日期范围时效率低下且易出错。本教程将介绍如何利用php内置的datetime类,简洁高效地解析日期字符串,并直接提取或格式化出月份信息,从而避免复杂的日期区间比较,…

    2025年12月13日
    000
  • PHP PDO实现用户密码条件更新:当输入为空时不修改密码

    本教程详细阐述了在php pdo应用中,如何实现用户密码的条件式更新。当用户在更新表单中未输入新密码时,系统将保留数据库中已有的密码,避免不必要的修改。文章通过优化sql的`if`语句,结合php的输入处理和安全实践,提供了一个健壮且高效的解决方案,同时纠正了常见的语法错误,确保数据更新的灵活性和准…

    2025年12月13日
    000
  • Laravel中根据用户认证状态和角色动态控制页面元素显示

    本文将详细介绍在laravel应用中如何根据用户的认证状态和特定角色动态控制页面元素的显示。针对访客、特定用户类型以及排除特定角色的用户,我们将通过结合`auth()->check()`和`auth()->user()`方法,提供一个健壮的解决方案,避免因未认证用户访问用户属性而导致的错…

    2025年12月13日
    000
  • 怎么测试php源码_用PHP环境测试源码功能教程【教程】

    验证PHP源码功能需四步:一、用XAMPP/WAMP搭建本地环境,启动Apache后通过http://localhost访问;二、使用3v4l.org等在线平台粘贴代码运行;三、在VS Code中配置PHP插件与launch.json进行断点调试;四、命令行输入php 文件名.php直接执行。 如果…

    2025年12月13日
    000
  • Laravel 中构建愿望清单:解决 foreach() 类型错误与数据存储策略

    在 Laravel 中使用 Cookie 构建愿望清单时,常见的问题是 `foreach()` 类型错误,这通常是由于将多个商品 ID 错误地存储为单个字符串或整数,导致 `Cookie::get()` 返回非数组类型数据。本文将深入分析此问题,并提供两种解决方案:优先推荐使用数据库存储多项愿望清单…

    2025年12月13日
    000
  • PHP PDO 条件更新密码字段的教程

    本教程详细阐述了在php pdo应用中,如何实现用户密码的条件更新。当用户在表单中未输入新密码时,系统应保留数据库中现有密码;反之,若输入了新密码,则进行更新并安全地哈希。文章将重点介绍使用sql的`if`函数来处理这种条件逻辑,并纠正常见的sql语法错误,确保数据更新的准确性和安全性,同时强调使用…

    2025年12月13日
    000
  • 理解服务器Ping与PHP脚本记录:区分ICMP与HTTP请求及其监控方案

    %ignore_a_1%脚本无法直接记录服务器的icmp ping请求,因为ping操作在操作系统内核的网络层处理,而非php运行的应用层。本文将阐明icmp ping与http请求的区别,解释php脚本的职责范围,并提供针对http访问记录和服务器可用性监控的正确方法及替代方案。 在服务器管理和网…

    2025年12月13日
    000

发表回复

登录后才能评论
关注微信