概念
- 按值传参:
函数的形参是被调用是实参的副本,修改形参的值并不会改变实参的值。 - 引用传参:
函数的形参接收实参的隐式引用,而不再是副本。修改形参的值这个时候会改变实参的值,因为他们指向相同的值。 - 共享传参:
js中基本类型是按值传递,对象类型按共享传递。调用函数传参时,函数接受对象实参引用的副本。
基本类型的传递
function add(num){
num=num+10;
console.log(num);//20
}
var number=10;
add(10);
console.log(number);//10
number值复制了一份给函数形参,函数内部形参的改变不会影响外部实参。
引用类型的传递
var obj={
value:1
};
function foo(o){
o.value=2;
console.log(o.value);//2
console.log(o);//{value: 2}
}
foo(obj);
console.log(obj);//{value: 2}
console.log(obj.value);//2
通过传参,形参的改变影响了外面的实参,那么这是不是就是引用传参呢?请继续看下面的示例:
var obj1={
value:1
}
function bar(o){
o=2;
console.log(o);//2
}
bar(obj1);
console.log(obj1);//{value: 1}
console.log(obj1.value);//1
如果是引用传递,那么在外面输出的obj1为什么还是{value:1}呢?在函数中,我们不是已经改变了吗?这也就说明js中引用类型进行参数的传递并不是按照引用传参,而是共享传参!
还可以这样解释:在向参数传递引用类型的值时,会把这个值在内存中的地址复制给局部变量,因此这个局部变量的变化会反应在函数外。也就是说引用类型把在内存中的地址复制给了函数内的局部变量,会出现两种情况:
- 改变引用类型的属性
当在函数中改变引用类型(如对象)的属性时,是在同一个内存区域进行操作。所以会在函数外反应出来。 - 列表项目
在函数内,对形参进行了重新赋值,即改变了形参的引用,这个时候内部的内存地址发生了改变,与实参引用已经完全不一样了,所以不会对函数外引用类型变量参数产生影响。