JS传参是按值传递还是按引用传递?

首先要清楚的是,什么是按值传递?什么又是按引用传递?
值传递(call by value):方法调用时,实际参数把它的值传递给对应的形式参数,形式参数只是用实际参数的值初始化自己的存储单元内容,是两个不同的存储单元,所以方法执行中形式参数值的改变不影响实际参数的值。
引用传递(call by refrence):也称为传地址。方法调用时,实际参数是对象(或数组),这时实际参数与形式参数指向同一个地址,在方法执行中,对形式参数的操作实际上就是对实际参数的操作,这个结果在方法结束后被保留了下来,所以方法执行中形式参数的改变将会影响实际参数。

两种传递方式都有各自的问题:按值传递由于每次都需要克隆副本,对一些复杂类型,性能较低;按引用传递会使函数调用的追踪更加困难,有时也会引起一些微妙的BUG。

基本类型按值传递

基本类型包括:Undefined,Null,Boolean,Number、String
下面直接看例子:

 

var a = 1;
function foo(val) {
  val = 2;
}
foo(a);
console.log(a);  // 输出结果为:1

分析以上代码,a是基本类型,存储在栈中,函数foo的的形参只是变量a的拷贝,调用函数时会在栈中新建一个变量val,并将变量a的值赋值给变量val,但是变量val和变量a并无关联,所以对变量val的修改操作并不会影响到变量a。

引用类型传递方式

引用类型包括:Object,Array,Date,RegExp,Function,......
我们先看下面的代码:

 

var person = {name:'MJ'};
function foo(obj) {
  obj.name = 'EP';
}
foo(person);
console.log(person.name); // 输出结果为:EP

对象person被修改了!这说明obj跟person指向的是同一个对象,可以肯定不是按值传递的了,那么就是引用传递了吗?从这个例子来看确实很像,我们再对上面的例子做下修改,如下:

 

var person = {name: "MJ"};
function foo(obj) {
  obj = new Object(); // 修改部分
  obj.name = "EP";
}
foo(person);
console.log(person.name); // 输出结果为:MJ

输出结果是MJ,也就是说person对象并未被修改,如此可以确定,js的引用类型也不是按引用传递。既不是按值传递,又不是按引用传递,那是按什么传递呢?

按共享传递(call by sharing):

准确的说,JS中的基本类型按值传递,引用类型按共享传递(call by sharing,也叫按对象传递、按对象共享传递)。最早由Barbara Liskov. 在1974年的GLU语言中提出。该求值策略被用于Python、Java、Ruby、JS等多种语言。
该策略的重点是:调用函数传参时,函数接受对象实参引用的副本(既不是按值传递的对象副本,也不是按引用传递的隐式引用)。 它和按引用传递的不同在于:在共享传递中对函数形参的赋值,不会影响实参的值。
接下来我们分析下上面的代码,为什么person没有被修改?
我们都知道对象保存在堆中,person持有堆中对象的引用,在调用foo函数时,函数接收的实际是person持有的对象(下面称之为原对象)引用的副本,这样obj和person都指向了同一个地方(也就是原对象在堆中存放的地址),而函数中的obj = new Object()操作其实是在堆中创建了一个新对象,并让obj持有该对象的引用,这样obj与原对象之间的关系就断开了,转而与新对象建立了关系,所以对obj的修改并不会影响到原对象。


总结

在JS中,基本类型按值传递,引用类型按共享传递

 

<think>好的,我现在需要帮用户理解函数传参的三种方式:值传递、地址传递引用传递的区别和联系。首先,我得回忆一下这些概念的基本定义,然后结合用户提供的三个引用内容来组织回答。 首先,用户提供的三个引用分别来自C++和JavaScript,所以可能需要区分不同语言中的实现差异。比如,JS中只有值传递,但如果是对象的话,传递的是引用的副本,这可能和C++的引用传递不同。不过用户主要问的是三种方式,所以重点可能在C++中的情况。 接下来,我需要明确每种传递方式的机制: 1. **值传递**:函数接收的是实参的拷贝。修改形参不会影响实参。适用于基本类型,但对于大对象可能效率低,且有浅拷贝问题。引用[2]提到值传递是单向的,形参是实参的拷贝,改变形参不影响实参。 2. **地址传递(指针传递)**:传递的是变量的地址,函数通过指针间接访问和修改实参。需要解引用操作,如引用[1]中的例子,用指针来交换变量。这种方式可以修改实参,但操作相对繁琐,需要注意指针的有效性。 3. **引用传递**:C++中的引用是别名,传递时直接操作原变量,语法更简洁。引用[2]提到引用传递不需要解引用,直接操作原变量,适合结构体和类,避免拷贝开销。 然后,需要比较它们的区别和联系: - **参数类型**:值传递是拷贝,地址传递是指针,引用传递是别名。 - **操作方式**:地址传递需要解引用引用传递直接使用。 - **效率**:值传递对于大对象效率低,地址和引用传递更高效。 - **安全性**:指针可能为空或野指针,引用则必须初始化且不为空。 - **语言差异**:如JS中的对象传递引用副本,属于共享传递,修改属性会影响原对象,但重新赋值不会,这点可能需要说明。 用户可能混淆不同语言中的表现,比如JS的“引用传递”其实和C++的不同。不过用户的问题可能更关注C++的情况,因为引用[1]和[2]都是C++相关的。但引用[3]提到JS的情况,可能需要简要对比。 然后,组织回答的结构: 1. 分点解释三种传递方式,每种包括定义、特点、示例代码(可能的话)、适用场景。 2. 对比三者的区别,比如参数类型、内存操作、效率、安全性。 3. 总结联系,比如地址和引用传递都能修改实参,值传递保护原数据。 4. 添加相关问题,帮助用户深入理解。 需要注意的是,用户提供的引用中有提到地址传递在C++中的例子,如交换函数使用指针,而引用传递则更简洁。同时,引用[2]指出复合类型避免值传递,推荐指针或引用。这些点都要整合到回答中。 可能需要检查示例代码的正确性,比如C++中的swap函数使用指针和引用的不同写法,确保语法正确。此外,是否需要区分指针传递引用传递在函数定义和调用时的差异,比如调用时指针需要取地址,而引用直接传递变量。 最后,确保回答符合用户的要求:结构清晰,分步骤解释,引用标记正确,添加相关问题。同时使用中文,数学公式正确用$符号。</think>### 函数传参的三种方式及区别 #### 一、值传递 **定义**:将实参的**值拷贝**一份传递给形参,函数内操作的是拷贝后的数据,不会影响原始实参。 **特点**: - 形参是实参的独立副本,修改形参不影响实参[^2]。 - 适用于基本数据类型(如`int`, `char`)或不需要修改实参的场景。 - 对复合类型(如结构体、类)可能引发性能问题(深拷贝开销)。 **示例**(C++): ```cpp void mySwapValue(int a, int b) { int temp = a; a = b; b = temp; // 交换失败,实参不变 } ``` --- #### 二、地址传递(指针传递) **定义**:将实参的**内存地址**传递给形参(指针),通过解引用操作直接修改内存中的原始数据。 **特点**: - 需使用取地址符`&`传递参数,函数内用`*`操作指针指向的值[^1]。 - 可修改实参,但需注意空指针或野指针问题。 - 适用于需要修改实参或传递大型数据的场景。 **示例**(C++): ```cpp void mySwapPointer(int* a, int* b) { int temp = *a; *a = *b; *b = temp; // 交换成功,实参被修改 } ``` --- #### 三、引用传递 **定义**:形参是实参的**别名**,直接操作原始数据,语法更简洁(无需解引用)。 **特点**: - 形参与实参共享内存地址,修改形参即修改实参。 - 语法安全(引用必须初始化且不可为空)。 - 适用于需要高效修改实参的场景,尤其是复合类型(如类、结构体)。 **示例**(C++): ```cpp void mySwapRef(int &a, int &b) { int temp = a; a = b; b = temp; // 交换成功,实参被修改 } ``` --- #### 四、三者的区别与联系 | **特性** | 值传递 | 地址传递 | 引用传递 | |----------------|----------------------|-----------------------|-----------------------| | **参数类型** | 拷贝值 | 指针 | 别名 | | **内存操作** | 独立副本 | 直接操作原数据 | 直接操作原数据 | | **效率** | 低(复合类型) | 高 | 高 | | **安全性** | 高(保护原数据) | 需防指针异常 | 高(不可为空) | | **语法复杂度** | 简单 | 需`*`和`&`操作符 | 简洁(无符号) | **联系**: - 地址传递引用传递都能直接修改实参,但引用传递语法更安全、简洁。 - 值传递通过拷贝保护原始数据,适用于函数内局部修改。 --- #### 五、跨语言对比(JavaScript) 在JavaScript中: - **基本类型**(如`number`, `string`)是值传递,修改形参不影响实参。 - **引用类型**(如`Object`, `Array`)是“共享传递”:传递的是引用的副本,修改属性会影响原对象,但重新赋值形参不会影响实参[^3]。 **示例**: ```javascript function updateObj(obj) { obj.value = 2; // 修改属性,原对象被影响 obj = { value: 3 }; // 重新赋值,原对象不变 } ``` ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值