形参和实参的问题

昨天编码的时候遇到函数的形参是指针问题,各种没搞懂,今天整理一下:


1.函数的参数分为形参和实参两种。在本小节中,进一步介绍形参、实参的特点和两者的关系。形参出现在函数定义中,在整个函数体内都可以使用,离开该函数则不能使用。实参出现在主调函数中,进入被调函数后,实参变量也不能使用。形参和实参的功能是作数据传送。发生函数调用时,主调函数把实参的值传送给被调函数的形参从而实现主调函数向被调函数的数据传送。
函数的形参和实参具有以下特点:
1. 形参变量只有在被调用时才分配内存单元,在调用结束时,即刻释放所分配的内存单元。因此,形参只有在函数内部有效。函数调用结束返回主调函数后则不能再使用该形参变量。
2. 实参可以是常量、变量、表达式、函数等,无论实参是何种类型的量,在进行函数调用时,它们都必须具有确定的值,以便把这些值传送给形参。因此应预先用赋值,输入等办法使实参获得确定值。
3. 实参和形参在数量上,类型上,顺序上应严格一致,否则会发生类型不匹配”的错误。
4. 函数调用中发生的数据传送是单向的。即只能把实参的值传送给形参,而不能把形参的值反向地传送给实参。 因此在函数调用过程中,形参的值发生改变,而实参中的值不会变化。


2.虽然函数是在被调用的时候,给形参赋值,但是这种赋值并不是说在函数调用的时候,形参就是实参,而是重新分配一个空间,存储函数要用的参数,这个参数的值等于实参的值。

在c语言中,函数调用都是传值。也就是说,形式参数是实际参数的一个备份。
  所以当形参是普通变量的时候,在函数中操作形参的值,函数返回后变成实参的地址所指向的值,两个内存不同,所存的值也不同。
      
当形参是指针的时候,copy过来的是指针的值,但是这个值所指向的内容是实参指向的内容,故改变之后实参的内容也改变了,但是这里改变指针的指向,将指针指向
另一块内存,这个时候函数结束,实参的内存还是没有变化。

  也就是说,数据实际都是存在内存中的,只有改变内存中的数据才能改变实际的值,所以形参必须和实参的内存一致,才能改变实参。

### C语言中形参实参的概念及区别 #### 一、形参实参的定义 在C语言中,函数调用涉及两个重要概念——形参(形式参数)实参(实际参数)。 - **形参**是指在函数声明或定义使用的变量名,用于接收外部传入的数据。这些变量仅存在于函数内部,在函数执行完毕后会被销毁[^1]。 - **实参**是在调用函数传递给函数的具体数据值或表达式。它通常是一个已知的变量或者常量。 例如: ```c #include <stdio.h> void func(int x) { printf("Value inside function: %d\n", x); } int main() { int a = 5; func(a); // 调用func函数并传递a作为实参 return 0; } ``` 在此代码片段中,`x` 是形参,而 `a` 则是实参[^2]。 --- #### 二、形参实参类型及其内存占用 C语言支持多种类型的参数传递,包括但不限于基本数据类型指针类型以及结构体类型等。每种类型都有其特定的字节大小用途: 1. **基本数据类型** 基本数据类型如 `int`, `float`, `char` 等具有固定的存储空间需求。例如,`int` 类型一般占据 4 字节(具体取决于编译器环境),而 `char` 占据 1 字节[^1]。 2. **指针类型** 指针是一种特殊的变量,用来保存另一个变量的地址。无论指针指向何种类型的数据,它的本身大小通常是相同的(比如在大多数现代架构下为 8 字节)。通过指针可以实现对原始数据的操作[^3]。 3. **结构体类型** 结构体允许将不同类型的数据组合在一起形成复合数据类型。当作为一个整体传递,整个结构体会被复制到对应的形参位置上[^1]。 --- #### 三、形参实参的传递方式 C语言提供了两种主要的参数传递方法:**值传递** **引用传递**。 1. **值传递** 当采用值传递的方式,会将实参的内容拷贝一份副本交给形参使用。这意味着即使修改了形参也不会影响原实参的值[^3]。 示例: ```c #include <stdio.h> void modifyValue(int num) { num *= 2; // 修改的是num的一个本地副本 printf("Inside Function: %d\n", num); } int main() { int value = 10; modifyValue(value); printf("After Call: %d\n", value); // 输出仍为原来的value return 0; } ``` 2. **引用传递** 如果希望函数能够直接影响实参,则可以通过指针来间接访问该对象。此传递的是指向实参指针而非其实质内容。 示例: ```c #include <stdio.h> void modifyByPointer(int* ptrNum) { (*ptrNum) *= 2; // 解引用操作符(*)作用于ptrNum所指向的位置 printf("Modified Inside Func via Pointer: %d\n", *ptrNum); } int main() { int number = 7; modifyByPointer(&number); // 将address-of(number)发送过去 printf("Final Value After Modification Using Pointers: %d\n", number); return 0; } ``` --- #### 四、总结 综上所述,理解区分好形参实参对于编写高效可靠的程序至关重要。需要注意的关键点有以下几个方面: - 形参只在其所属的作用域范围内有效; - 实参决定了最终如何初始化相应的形参; - 不同情况下可以选择不同的传递策略以满足功能需求。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值