JavaScript的参数传递方式:传值or传引用

标签 IT工具网 javascript

原问题:Is JavaScript a pass-by-reference or pass-by-value language?

一直没仔细去想过JavaScript中函数参数的传递方式,直到在上个项目中遇到过几次参数是以引用方式传递的情况之后,才弄清楚了。借这个问题总结一下。

现在排名第一的答案总结得很精简,我以它来说明。

传值还是传引用?

function changeStuff(num, obj1, obj2)
{
    num = num * 10;
    obj1.item = "changed";
    obj2 = {item: "changed"};
}

var num = 10;
var obj1 = {item: "unchanged"};
var obj2 = {item: "unchanged"};
changeStuff(num, obj1, obj2);
console.log(num);   // 10
console.log(obj1.item);    // changed
console.log(obj2.item);    // unchanged

首先得澄清一下大家常说的传值和传引用。仍然以上面的changeStuff(num, obj1, obj2)为例。

传值

函数内的num, obj1, obj2都将是一份新的内存,与调用函数之前定义的三个变量毫无关系。函数内无论怎么修改这三个参数,外部定义的三个变量的值始终不变

传值的意思就是:传内存拷贝。

传引用

函数内的num, obj1, obj2都分别指向一块内存,该内存就是调用函数之前定义的三个变量时创建的内存。函数内对这三个参数所做的任何改动,都将反映到外部定义的三个变量上。

传引用的意思就是:传内存指针。

从上面的代码可以看出,JavaScript中函数参数的传递方式既不是传值,也不是传引用,而是叫call-by-sharing

Call-by-sharing

它的意思是:传引用的拷贝。

  • changeStuff()中的num, obj1, obj2都是一个引用
  • 他们的内容是某块内存的地址
  • 这个地址的值来自于外部定义的三块内存,是那三块内存地址的一份拷贝
  • 同时,在函数内部这三个参数的值是可以直接被修改的,可以指向其他对象(由于JavaScript中没有指针或引用运算符,只能直接修改)

因此,我们从内存和引用的角度再来看看changeStuff()的定义:

function changeStuff(num, obj1, obj2)
{
    num = num * 10; // 对num赋值,修改num的指向,新内存的内容为old_num * 10
    obj1.item = "changed";  // 修改原始obj1内存中的内容
    obj2 = {item: "changed"};   // 对obj2赋值,修改obj2指向,新内存的内容为{item: "changed"}
}

这里我们忽略num = num * 10具体是如何完成的。这依赖于具体语言的解释器,实际上包含了数个对指针的操作。

由于call-by-sharing本质上也是传值,因此,也可以说JavaScript的传参方式都是传值。

除了JavaScript之外,Python, Java, Ruby, Scheme等语言也是采用call-by-sharing的求值策略。

相关文章:

java finally代码块总会被执行么

java - 实现Runnable接口(interface) 和 继承Thread类

Java如何遍历map对象

javascript - 正在寻找稳定的 js-Tree PlugIn、jsTree 或 Ext-Js Tree?

javascript - 在 Web 应用程序上使用 Adob​​e Flash 图表库是个好主意吗?

Git如何更新密码,fatal: Authentication failed for

Java中软引用和弱引用的区别

javascript - 在扩展的内容脚本中使用 document.write 加载脚本

c# - 使用邮政编码条目自动填充城市和州

javascript - javascript中的转义点