乐趣区

复制和引用复制

在编程语言中,赋值和参数传递可以通过 value-copy(值复制)或者 reference-copy(引用复制)来完成,这取决于使用的是什么语法。
C 语言中有指针这个概念,如果要向函数传递一个数并在函数中更改它的值,可以通过‘&’传递变量,这就是 reference-copy;若是没有声明为引用的话,就是通过 value-copy 的方式传递。
但在 JavaScript 中没有指针这个概念,但也有这两种复制,不过,在语法上没有区别,并且引用指向的是值。这两种复制完全根据值的类型来决定。

// 第一种
let a = 2;
let b = a; // b 是 a 的值一个副本,也就是 value-copy
b++;
a; //2
b; //3
// 第二种
let c = [1, 2, 3];
let d = c; // d 是 [1, 2, 3] 的一个引用,引用指向的是值哦,这是 reference-copy
d.push(4);
c; // [1, 2, 3, 4]
d; // [1, 2, 3, 4]
由上可知,基本类型的值是通过值复制的方式来赋值或是传递的,基本类型有 null、undefined、字符串、数字、布尔以及 ES6 中的 symbol。引用类型的值是通过引用复制的方式来赋值或是传递的,引用类型有数组、对象、函数等。由于引用指向的是值本身而非变量,所以一个引用无法更改另一个引用的指向。也就是说,多个引用相互之间没有引用 / 指向关系。关于这点,看代码就清楚了,如下:
let c = [1, 2, 3];
let d = c; // 这时 d 和 c 都为[1, 2, 3]

d = [4, 5, 6];
c; // [1, 2, 3]
d; // [4, 5, 6]
// d = [4, 5, 6]; 这句赋值语句并不会影响 c 指向值[1, 2, 3], 除非 b 是指向 c 的,但上面我们已经说过‘引用指向的是值本身而非变量’!!!
// 要知道直接赋值和 push()是不一样的,前者是重新引用,而后者是在引用的基础上往里面加东西。
// 若是想要在引用的基础上清空数组,可以用 arr.length = 0 的方式

特别要注意的一点是,那些基本类型,在初始化时,可以封装成对象,可就算如此操作,还是更改不了‘它是基本类型值’的事实,所以还是 value-copy,代码如下:
function add(x) {
x += 1;
}
let a = new Number(1); // typeof a => object
add(a);
a; // a 为 2,而不是 3

退出移动版