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;
10
a
a
10
而当我们声明一个对象变量,比如
let obj = { name: "Alice" };{ name: "Alice" }obj
所以,当
obj
在日常开发中,参数传递带来的“意外”通常是指函数内部对对象参数的修改,不经意间影响了函数外部的原始对象,导致了难以追踪的副作用。要避免这种问题,有几个策略非常实用:
明确意图:是否需要修改原对象? 在编写函数时,首先要问自己:这个函数是否应该修改传入的对象?如果答案是“不应该”,那么就必须采取措施来保护原始对象。如果答案是“应该”,那么就需要在函数签名或文档中明确指出这一点,让调用者知道函数会产生副作用。
创建浅拷贝(Shallow Copy) 当你只需要修改对象的第一层属性,且不希望影响原始对象时,浅拷贝是一个简单有效的办法。
Object.assign({}, originalObject){...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_.cloneDeep()
采用函数式编程思想:纯函数 尽量编写“纯函数”(Pure Function)。纯函数有两个特点:
通过这些实践,我们不仅能更清晰地理解 JavaScript 的参数传递机制,也能在实际开发中写出更健壮、更可维护的代码。毕竟,代码的健壮性往往比一时的“简洁”更重要。
以上就是JS 函数参数传递机制 - 值传递与引用传递的误解与真相剖析的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号