答案相当简单:因为 ECMAScript 是按值传递而不是不是按引用传递,而您的代码证明了这一点。 (更准确地说,是call-by-sharing,是一种特殊的传值方式。)
请参阅Is JavaScript a pass-by-reference or pass-by-value language? 了解更多信息。
ECMAScript 使用按值传递,或者更准确地说,是按值传递的一种特殊情况,其中传递的值始终 是一个指针。这种特殊情况有时也称为 call-by-sharing、call-by-object-sharing 或 call-by-object。
Java(用于对象)、C#(默认情况下用于引用类型)、Smalltalk、Python、Ruby 以及几乎所有创建的面向对象语言都使用相同的约定。
注意:某些类型(例如)Numbers 实际上是通过值直接传递的,而不是通过中间指针传递的。但是,由于它们是不可变的,在这种情况下,按值传递和按对象共享之间没有可观察到的行为差异,因此您可以通过简单地处理所有内容来大大简化您的心智模型作为按对象共享调用。只需将这些特殊情况解释为您无需担心的内部编译器优化即可。
这是一个简单的示例,您可以运行它来确定 ECMAScript(或任何其他语言,在您翻译后)的参数传递约定:
function isEcmascriptPassByValue(foo) {
foo.push('More precisely, it is call-by-object-sharing!');
foo = 'No, ECMAScript is pass-by-reference.';
return;
}
var bar = ['Yes, of course, ECMAScript *is* pass-by-value!'];
isEcmascriptPassByValue(bar);
console.log(bar);
// Yes, of course, ECMAScript *is* pass-by-value!,
// More precisely, it is call-by-object-sharing!
如果你熟悉 C#,这是了解值类型和引用类型的值传递和引用传递之间的区别的一个很好的方法,因为 C# 支持所有 4 种组合:值类型的值(“传统的值传递”),引用类型的值传递(共享调用、对象调用、对象共享调用,如 ECMAScript 中)、传递-引用类型的按引用,值类型的按引用。
(实际上,即使您不了解 C#,这也不难。)
struct MutableCell
{
public string value;
}
class Program
{
static void IsCSharpPassByValue(string[] foo, MutableCell bar, ref string baz, ref MutableCell qux)
{
foo[0] = "More precisely, for reference types it is call-by-object-sharing, which is a special case of pass-by-value.";
foo = new string[] { "C# is not pass-by-reference." };
bar.value = "For value types, it is *not* call-by-sharing.";
bar = new MutableCell { value = "And also not pass-by-reference." };
baz = "It also supports pass-by-reference if explicitly requested.";
qux = new MutableCell { value = "Pass-by-reference is supported for value types as well." };
}
static void Main(string[] args)
{
var quux = new string[] { "Yes, of course, C# *is* pass-by-value!" };
var corge = new MutableCell { value = "For value types it is pure pass-by-value." };
var grault = "This string will vanish because of pass-by-reference.";
var garply = new MutableCell { value = "This string will vanish because of pass-by-reference." };
IsCSharpPassByValue(quux, corge, ref grault, ref garply);
Console.WriteLine(quux[0]);
// More precisely, for reference types it is call-by-object-sharing, which is a special case of pass-by-value.
Console.WriteLine(corge.value);
// For value types it is pure pass-by-value.
Console.WriteLine(grault);
// It also supports pass-by-reference if explicitly requested.
Console.WriteLine(garply.value);
// Pass-by-reference is supported for value types as well.
}
}