前言:本文主要介绍了C语言中实参和形参的区别详解,文中通过示例代码介绍的非常详细,对大家的学习具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!
一、基本概念
1. 实际参数 (实参) :
真实传给函数的参数,叫实参。
实参可以是:常量、变量、表达式、函数等。
无论实参是何种类型的量,在进行函数调用时,它们都必须有确定的值,以便把这些值传送给形参。
2. 形式参数 (形参) :
形式参数是指函数名后括号中的变量,因为形式参数只有在函数被调用的过程中才实例化(分配内存单元),所以叫形式参数。形式参数当函数调用完成之后就自动销毁了。因此形式参数只在函数 中有效,与局部变量的生命周期意义相似,可以对比理解。
二、代码示例
我们举一个例子:
写一个函数可以交换两个整形变量的内容。
分析题干,我们知道交换两个整形变量的内容,需要我们引入一个中间变量 t ,首先将第一个整形变量的值赋给中间变量 t ,接着将第二个整形变量的值赋给第一个整形变量,最后将中间变量的值赋给第二个中间变量。即可实现上述功能。代码如下:
#include <stdio.h>
//实现成函数,但是不能完成任务
void Swap1(int x, int y)
{
int t = 0;
t = x;
x = y;
y = t;
}
//正确的版本
void Swap2(int* px, int* py)
{
int t = 0;
t = *px;
*px = *py;
*py = t;
}
int main()
{
int num1 = 20;
int num2 = 100;
printf("交换前:num1 = %d num2 = %d\n", num1, num2);
Swap1(num1, num2);
printf("Swap1:num1 = %d num2 = %d\n", num1, num2);
Swap2(&num1, &num2);
printf("Swap2:num1 = %d num2 = %d\n", num1, num2);
return 0;
}
代码运行结果如下:
上面 Swap1 和 Swap2 函数中的参数 x,y,px,py 都是形式参数。
在main函数中传给 Swap1 的 num1 ,num2 和传给 Swap2 函数的 &num1 , &num2 是实际参数。
那么为什么Swap1函数无法实现交换变量的功能呢?
在这里我们对函数的实参和形参进行分析:
我们可以发现实参num1,num2和形参x,y使用的不是同一内存空间。
当 num1 和 num2 传入 Swap1() 函数的时候,x 和 y 的确可以拿到 100 和 20 这两个值,但 x 和 y 本身与 num1 和 num2 分别具有不同的地址,这就造成一个错误:在Swap1() 函数中,只交换了形参 x 和 y 的值,对于实参 a 和 b 没有任何影响,因此输出结果时无法实现交换变量值的功能。
这里可以看到 Swap1 函数在调用的时候, x , y 拥有自己的空间,同时拥有了和实参一模一样的内容。 所以我们可以简单的认为:形参实例化之后其实相当于实参的一份临时拷贝。
那为什么Swap2函数无法实现交换变量的功能呢?
传入Swap2函数的是实参num1,num2的地址,而非实参num1,num2的值。在Swap2函数中,形参px,py用整形指针来接收实参num1,num2的地址,再通过解引用符号' * '来拿到对应地址的值,最后通过中间变量 t 完成交换。
需要注意的是 :px,py是用于存放num1,num2地址的指针变量,本身也有自己的地址,只是用于存储实参的地址。由图也可以看出,二者地址并不相同,不要搞混了。
Swap2函数本质上采用了传址调用。不同于Swap1函数采用的传值调用,这种传参方式可以让函数和函数外部的变量建立起真正的联系,函数内部可以直接操作外部的变量。
三、总结
总的来说,面对不同的问题,需要我们灵活选择传值调用或是传址调用。
通过对上述代码的分析,我们可以从中总结出以下两点:
//当实参传递给形参的时候,形参是实参的一份临时拷贝。
//对形参的修改不影响实参。