JS 函数参数传递机制 – 值传递与引用传递的误解与真相剖析

JavaScript函数参数传递本质是值传递,原始类型传值副本,对象类型传引用地址副本,因此修改对象属性会影响外部对象,但重新赋值参数不影响。

js 函数参数传递机制 - 值传递与引用传递的误解与真相剖析

JavaScript 的函数参数传递机制,核心就一句话:它永远是值传递。无论是原始类型(如数字、字符串)还是对象类型(包括数组、函数),传递的都是变量的值。只不过,对于对象而言,这个“值”是一个指向内存中实际对象的引用(或者说内存地址),而不是对象本身。所以,我们看到的“引用传递”现象,本质上是“引用地址的值传递”。

解决方案

理解 JavaScript 参数传递的关键在于区分原始值和引用值。当我们把一个变量作为参数传递给函数时,JavaScript 会创建这个参数的一个局部副本。

对于原始值类型(

number

,

string

,

boolean

,

null

,

undefined

,

symbol

,

bigint

),这个局部副本就是原始值本身的一个精确拷贝。这意味着,你在函数内部对这个参数的任何修改,都只会影响到这个局部副本,而不会触及到函数外部的原始变量。

function modifyPrimitive(num) {  num = num + 10;  console.log("Inside function (primitive):", num); // 20}let myNumber = 10;modifyPrimitive(myNumber);console.log("Outside function (primitive):", myNumber); // 10 (unaffected)

而对于对象类型(

Object

,

Array

,

Function

),情况就显得有些微妙了。此时传递的“值”并不是对象本身,而是存储在变量中的那个指向对象内存地址的“引用”。函数接收到的是这个引用地址的一个副本。这意味着,函数内部的参数和外部的原始变量现在都指向内存中的同一个对象。因此,如果你通过这个引用去修改对象的属性,那么外部的原始对象也会随之改变,因为它们指向的是同一块内存区域。

function modifyObjectProperties(obj) {  obj.name = "Jane Doe";  obj.age = 30;  console.log("Inside function (object properties):", obj); // { name: 'Jane Doe', age: 30 }}let myObject = { name: "John Doe", age: 25 };modifyObjectProperties(myObject);console.log("Outside function (object properties):", myObject); // { name: 'Jane Doe', age: 30 } (affected)

然而,如果你在函数内部尝试将参数重新赋值为一个全新的对象,那么这个操作只会改变函数内部局部参数的指向,使其指向新的内存地址,而不会影响到函数外部的原始变量。因为原始变量依然持有它最初的那个引用地址。

function reassignObject(obj) {  obj = { city: "New York" }; // obj现在指向了一个全新的对象  console.log("Inside function (reassigned object):", obj); // { city: 'New York' }}let anotherObject = { country: "USA" };reassignObject(anotherObject);console.log("Outside function (reassigned object):", anotherObject); // { country: 'USA' } (unaffected)

这三个例子,在我看来,基本就涵盖了所有关于 JavaScript 参数传递的“真相”了。

为什么很多人会误解为“引用传递”?

说实话,这种误解真的太普遍了,甚至我刚开始接触 JavaScript 的时候也曾一度困惑。我想,这主要是因为当我们在函数内部修改了传入对象的属性时,函数外部的原始对象也确实发生了变化,这看起来简直就是“引用传递”的典型行为啊!在许多其他语言(比如 C++ 的引用)中,这种行为就是引用传递的标志。所以,把这种现象直接归结为“引用传递”,似乎是直观且“符合经验”的。

但问题在于,JavaScript 的“引用”和 C++ 里的“引用”是不同的概念。JavaScript 传递的那个“引用”,本身也是一个值。你可以把它想象成一个门牌号或者内存地址。当你把一个对象的门牌号传给函数时,函数拿到的只是这个门牌号的副本。函数内部和外部的变量,现在都有了同一个门牌号,所以它们都能找到同一栋房子。你通过门牌号副本去修改房子内部的装修,房子当然会变。但是,如果你在函数内部把这个门牌号副本换成了另一个新房子的门牌号,那只是你手里的门牌号变了,外面那个人手里的门牌号(指向老房子)可没变。

这种“看起来像引用传递,但实际是值传递”的机制,其实是 JavaScript 设计哲学中的一个权衡。它既提供了操作共享对象的能力,又避免了像 C++ 那样直接操作内存地址可能带来的复杂性和风险。理解这一点,就能避开很多潜在的坑。

深入理解:原始值与对象在内存中的存储差异

要彻底搞清楚参数传递,我们不得不稍微深入一点,看看 JavaScript 在内存里是怎么“玩”的。这背后其实是原始值和对象在内存中存储方式的根本区别

当我们声明一个原始值变量,比如

let a = 10;

,通常情况下(虽然现代 JavaScript 引擎有优化,但从概念上理解),这个

10

这个值是直接存储在变量

a

所占据的内存空间里的,我们称之为内存(Stack)。栈内存的特点是结构简单、存取速度快,但空间有限,主要用于存储原始值和函数调用的上下文。当你把

a

传递给函数时,函数参数会开辟一个新的内存空间,把

10

这个值复制一份放进去。两者之间再无瓜葛。

而当我们声明一个对象变量,比如

let obj = { name: "Alice" };

,事情就完全不同了。

{ name: "Alice" }

这个实际的对象数据,是存储在堆内存(Heap)中的。堆内存的特点是空间大、灵活,可以存储任意大小的复杂数据结构,但存取速度相对较慢。变量

obj

本身在栈内存中存储的,并不是对象的所有数据,而是一个指向堆内存中那个实际对象数据的“地址”(或者叫引用)。

所以,当

obj

作为参数传递给函数时,函数接收到的,正是这个“地址”的副本。函数内部的参数变量,现在也拥有了同一个地址。它们就像两把钥匙,都能打开同一扇门,进入同一间屋子。你通过其中一把钥匙进入屋子,改变了屋子里的家具摆设,另一把钥匙再进去时,看到的就是改变后的样子。但如果你用其中一把钥匙去配了一把新屋子的钥匙,那和老屋子就没关系了。这种内存模型,是理解所有 JavaScript 引用行为的基础。

实际开发中如何避免参数传递带来的“意外”?

在日常开发中,参数传递带来的“意外”通常是指函数内部对对象参数的修改,不经意间影响了函数外部的原始对象,导致了难以追踪的副作用。要避免这种问题,有几个策略非常实用:

明确意图:是否需要修改原对象?在编写函数时,首先要问自己:这个函数是否应该修改传入的对象?如果答案是“不应该”,那么就必须采取措施来保护原始对象。如果答案是“应该”,那么就需要在函数签名或文档中明确指出这一点,让调用者知道函数会产生副作用。

创建浅拷贝(Shallow Copy)当你只需要修改对象的第一层属性,且不希望影响原始对象时,浅拷贝是一个简单有效的办法。

对象: 使用

Object.assign({}, originalObject)

或 ES6 的展开运算符

{...originalObject}

function processUser(user) {  const newUser = { ...user }; // 创建一个浅拷贝  newUser.status = "active";  return newUser;}let userProfile = { id: 1, name: "Bob" };let activeUser = processUser(userProfile);console.log(userProfile); // { id: 1, name: 'Bob' } (未被修改)console.log(activeUser);  // { id: 1, name: 'Bob', status: 'active' }

数组: 使用

Array.prototype.slice()

或展开运算符

[...originalArray]

function addId(items) {  const newItems = [...items]; // 创建一个浅拷贝  newItems.push(newItems.length + 1);  return newItems;}let myArr = [10, 20];let updatedArr = addId(myArr);console.log(myArr);       // [10, 20] (未被修改)console.log(updatedArr);  // [10, 20, 3]

需要注意的是,浅拷贝只复制了对象的第一层。如果对象内部嵌套了其他对象,那么这些嵌套对象仍然是共享的引用。

创建深拷贝(Deep Copy)如果你的对象包含多层嵌套,并且你希望完全独立地操作这个对象,那么你需要进行深拷贝。

JSON.parse(JSON.stringify(object))

这是最常见也最简单的深拷贝方法,但它有局限性。它不能拷贝函数、

undefined

symbol

Date

对象(会变成字符串)、正则表达式等。对于只包含基本数据类型和普通对象的复杂数据结构,它通常足够用。

function processComplexObject(data) {  const newData = JSON.parse(JSON.stringify(data));  newData.details.age = 40;  return newData;}let complexObj = { name: "Charlie", details: { age: 35, city: "London" } };let processedObj = processComplexObject(complexObj);console.log(complexObj.details.age); // 35console.log(processedObj.details.age); // 40

第三方库: 对于更复杂的场景,例如需要拷贝函数、循环引用等,Lodash 的

_.cloneDeep()

等专业库是更好的选择。

采用函数式编程思想:纯函数尽量编写“纯函数”(Pure Function)。纯函数有两个特点:

给定相同的输入,总是返回相同的输出。不会产生任何副作用,即不修改函数外部的任何状态(包括传入的对象参数)。遵循这个原则,可以大大减少代码中的“意外”和 bugs。

通过这些实践,我们不仅能更清晰地理解 JavaScript 的参数传递机制,也能在实际开发中写出更健壮、更可维护的代码。毕竟,代码的健壮性往往比一时的“简洁”更重要。

以上就是JS 函数参数传递机制 – 值传递与引用传递的误解与真相剖析的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
怎么使用JavaScript操作浏览器缓存?
上一篇 2025年12月20日 14:39:11
高效获取GitHub用户列表:使用GitHub API进行分页与数据检索
下一篇 2025年12月20日 14:39:20

相关推荐

  • composer require-dev和require有什么不同_Composer Require与Require-Dev区别解析

    require用于声明项目运行必需的依赖,如框架、数据库组件和第三方SDK,这些包会随项目部署到生产环境;2. require-dev用于声明仅在开发和测试阶段需要的工具,如PHPUnit、PHPStan、Faker等,不会默认部署到生产环境;3. 安装时composer install根据环境决定…

    2026年5月10日
    1000
  • 修复Django电商项目中AJAX过滤产品列表图片不显示问题

    在Django电商项目中,当使用AJAX动态加载过滤后的产品列表时,常遇到图片无法正常显示的问题。这通常是由于前端模板中图片加载方式(如data-setbg属性结合JavaScript库)与AJAX动态内容更新机制不兼容所致。解决方案是直接在AJAX返回的HTML中使用标准的标签来渲染图片,确保浏览…

    2026年5月10日
    000
  • 开源免费PHP工具 PHP开发效率提升利器

    推荐开源免费PHP开发工具以提升效率:VS Code、Sublime Text轻量高效,PhpStorm专业强大;调试用Xdebug、Kint、Ray;依赖管理选Composer;代码质量工具包括PHPStan、Psalm、PHP_CodeSniffer;数据库管理可用%ignore_a_1%MyA…

    2026年5月10日
    000
  • Golang JSON序列化:控制敏感字段暴露的最佳实践

    本教程探讨golang中如何高效控制结构体字段在json序列化时的可见性。当需要将包含敏感信息的结构体数组转换为json响应时,通过利用`encoding/json`包提供的结构体标签,特别是`json:”-“`,可以轻松实现对特定字段的忽略,从而避免敏感数据泄露,确保api…

    2026年5月10日
    000
  • 比特币新手教程 比特币交易平台有哪些

    比特币是一种去中心化的数字货币,基于区块链技术实现点对点交易,具有匿名性、有限发行和不可篡改等特点;新手可通过交易所购买,P2P交易获得比特币,常用平台包括Binance、OKX和Huobi;交易流程包括注册账户、实名认证、绑定支付方式、充值法币并下单购买,可选择市价单或限价单;比特币存储方式有交易…

    2026年5月10日
    000
  • c++中的SFINAE技术是什么_c++模板编程中的SFINAE原理与应用

    SFINAE 是“替换失败不是错误”的原则,指模板实例化时若参数替换导致错误,只要存在其他合法候选,编译器不报错而是继续重载决议。它用于条件启用模板、类型检测等场景,如通过 decltype 或 enable_if 控制函数重载,实现类型特征判断。尽管 C++20 引入 Concepts 简化了部分…

    2026年5月10日
    000
  • Go语言mgo查询构建:深入理解bson.M与日期范围查询的正确实践

    本文旨在解决go语言mgo库中构建复杂查询时,特别是涉及嵌套`bson.m`和日期范围筛选的常见错误。我们将深入剖析`bson.m`的类型特性,解释为何直接索引`interface{}`会导致“invalid operation”错误,并提供一种推荐的、结构清晰的代码重构方案,以确保查询条件能够正确…

    2026年5月10日
    100
  • 修复点击时按钮抖动:CSS垂直对齐实践

    本文探讨了在Web开发中,交互式按钮(如播放/暂停按钮)在点击时发生意外垂直位移的问题。通过分析CSS样式变化对元素布局的影响,我们发现这是由于按钮不同状态下的边框样式和内边距改变,以及默认的垂直对齐行为共同作用所致。核心解决方案是利用CSS的vertical-align属性,将其设置为middle…

    2026年5月10日
    100
  • 理解编程指令:当结果正确,但实现方式不符要求时

    本文探讨了在编程实践中,即使程序输出了正确的结果,但若其实现方式未能严格遵循既定指令,仍可能被视为“不正确”的问题。我们将通过具体示例,对比直接求和与累加求和两种实现策略,强调理解和遵守编程规范的重要性,以确保代码的健壮性、可维护性及符合项目要求。 在软件开发过程中,我们经常会遇到这样的情况:编写的…

    2026年5月10日
    000
  • Golang goroutine与channel调试技巧

    使用go run -race检测数据竞争,结合runtime.NumGoroutine监控协程数量,通过pprof分析阻塞调用栈,利用select超时避免永久阻塞,有效排查goroutine泄漏、死锁和数据竞争问题。 Go语言的goroutine和channel是并发编程的核心,但它们也带来了调试上…

    2026年5月10日
    000
  • 使用 Jupyter Notebook 进行探索性数据分析

    Jupyter Notebook通过单元格实现代码与Markdown结合,支持数据导入(pandas)、清洗(fillna)、探索(matplotlib/seaborn可视化)、统计分析(describe/corr)和特征工程,便于记录与分享分析过程。 Jupyter Notebook 是进行探索性…

    2026年5月10日
    000
  • 《魔兽世界》将于6月11日开启国服回归技术测试

    《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试

    《%ign%ignore_a_1%re_a_1%》官方宣布,将于6月11日开启国服回归技术测试,时间为7天,并称可以在6月内正式开服,玩家们可以访问官网下载战网客户端并预下载“巫妖王之怒”客户端,技术测试详情见下图。 WordAi WordAI是一个AI驱动的内容重写平台 53 查看详情 以上就是《…

    2026年5月10日 用户投稿
    200
  • php常量怎么用_PHP常量(define/const)定义与使用方法

    PHP中可通过define函数和const关键字定义常量,用于存储不可变值。define适用于全局作用域,支持动态名称和条件定义,如define(‘SITE_NAME’, ‘MyWebsite’);const在编译时生效,语法简洁但限制多,只能在类或全…

    2026年5月10日
    000
  • 如何在HTML中插入表单元素_HTML表单控件与输入类型使用指南

    HTML表单通过标签构建,包含action和method属性定义数据提交目标与方式,常用input类型如text、password、email等适配不同输入需求,配合label、required、placeholder提升可用性,结合textarea、select、button等控件实现完整交互,是…

    2026年5月10日
    100
  • 前端缓存策略与JavaScript存储管理

    根据数据特性选择合适的存储方式并制定清晰的读写与清理逻辑,能显著提升前端性能;合理运用Cookie、localStorage、sessionStorage、IndexedDB及Cache API,结合缓存策略与定期清理机制,可在保证用户体验的同时避免安全与性能隐患。 前端缓存和JavaScript存…

    2026年5月10日
    200
  • c#文件怎么打开

    打开 C# 文件有三种方法:Visual Studio:启动 Visual Studio,通过“文件”菜单打开 C# 文件。文本编辑器:使用文本编辑器打开 C# 文件,将其视为普通文本。.NET Core 命令行工具:使用 csc.exe 命令行工具编译 C# 文件,生成可执行文件。 如何打开 C#…

    2026年5月10日
    000
  • HTML5网页如何实现手势操作 HTML5网页移动端交互的处理技巧

    首先利用原生touch事件实现滑动判断,再通过preventDefault解决滚动冲突,接着引入Hammer.js处理复杂手势,最后通过优化点击区域、避免事件冲突和增加视觉反馈提升体验。 在移动端浏览器中,HTML5网页可以通过触摸事件实现手势操作,提升用户体验。虽然原生JavaScript提供了基…

    2026年5月10日
    000
  • 创建指定大小并填充特定数据的Golang文件教程

    本文将介绍如何使用Golang创建一个指定大小的文件,并用特定数据填充它。我们将使用 `os` 包提供的函数来创建和截断文件,从而实现快速生成大文件的目的。示例代码展示了如何创建一个10MB的文件,并将其填充为全零数据。掌握这些方法,可以方便地在例如日志系统或磁盘队列等场景中,预先创建测试文件或初始…

    2026年5月10日
    000
  • 深入理解 Express.js 中 next() 参数的作用与中间件机制

    本文深入探讨 express.js 中间件函数中的 `next()` 参数。它负责将控制权传递给请求-响应周期中的下一个中间件或路由处理程序。文章将详细解释 `next()` 的工作原理、中间件的注册与执行顺序,以及不正确使用 `next()` 可能导致请求挂起的风险,并通过代码示例和实际应用场景,…

    2026年5月10日
    000
  • Python命令怎样使用profile分析脚本性能 Python命令性能分析的基础教程

    使用Python的cProfile模块分析脚本性能最直接的方式是通过命令行执行python -m cProfile your_script.py,它会输出每个函数的调用次数、总耗时、累积耗时等关键指标,帮助定位性能瓶颈;为进一步分析,可将结果保存为文件python -m cProfile -o ou…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信