JS函数参数传递是值传递还是引用传递?
IT行业相对于一般传统行业,发展更新速度更快,一旦停止了学习,很快就会被行业所淘汰。所以我们需要踏踏实实的不断学习,精进自己的技术,尤其是初学者。今天golang学习网给大家整理了《JS函数参数传递机制:值传递还是引用传递?》,聊聊,我们一起来看看吧!
JavaScript函数参数传递本质是值传递,原始类型传值副本,对象类型传引用地址副本,因此修改对象属性会影响外部对象,但重新赋值参数不影响。

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); // 35 console.log(processedObj.details.age); // 40- 第三方库: 对于更复杂的场景,例如需要拷贝函数、循环引用等,Lodash 的
_.cloneDeep()等专业库是更好的选择。
采用函数式编程思想:纯函数 尽量编写“纯函数”(Pure Function)。纯函数有两个特点:
- 给定相同的输入,总是返回相同的输出。
- 不会产生任何副作用,即不修改函数外部的任何状态(包括传入的对象参数)。 遵循这个原则,可以大大减少代码中的“意外”和 bugs。
通过这些实践,我们不仅能更清晰地理解 JavaScript 的参数传递机制,也能在实际开发中写出更健壮、更可维护的代码。毕竟,代码的健壮性往往比一时的“简洁”更重要。
到这里,我们也就讲完了《JS函数参数传递是值传递还是引用传递?》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于对象类型,引用传递,值传递,深拷贝,JS参数传递的知识点!
Linuxroot密码忘记怎么重置?
- 上一篇
- Linuxroot密码忘记怎么重置?
- 下一篇
- 办公语音转文字软件推荐-手机版十大神器盘点
-
- 文章 · 前端 | 7小时前 |
- CSSz-index层级控制全攻略
- 394浏览 收藏
-
- 文章 · 前端 | 7小时前 |
- PostCSS插件配置全攻略
- 258浏览 收藏
-
- 文章 · 前端 | 7小时前 | 背景 CSS渐变 linear-gradient radial-gradient 颜色停点
- CSS渐变色详解:linear-gradient与radial-gradient用法
- 402浏览 收藏
-
- 文章 · 前端 | 7小时前 | 主题切换 color属性 currentColor 颜色统一管理 减少重复代码
- CSScurrentColor统一颜色管理技巧
- 160浏览 收藏
-
- 文章 · 前端 | 8小时前 |
- CSS导入外部样式表方法详解
- 189浏览 收藏
-
- 文章 · 前端 | 8小时前 |
- WebCryptoAPI:JavaScript密码学实战教程
- 140浏览 收藏
-
- 文章 · 前端 | 8小时前 |
- JS对象属性变化监听全解析
- 310浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3193次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3405次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3436次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4543次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3814次使用
-
- JavaScript函数定义及示例详解
- 2025-05-11 502浏览
-
- 优化用户界面体验的秘密武器:CSS开发项目经验大揭秘
- 2023-11-03 501浏览
-
- 使用微信小程序实现图片轮播特效
- 2023-11-21 501浏览
-
- 解析sessionStorage的存储能力与限制
- 2024-01-11 501浏览
-
- 探索冒泡活动对于团队合作的推动力
- 2024-01-13 501浏览

