先来讲一下函数的组成。
void Swap1(int x, int y)
{
int tmp = x;
x = y;
y = tmp;
}
上面这串代码中void是函数的返回类型,这里的函数无需返回值回去,所以用void。当你需要返回值返回主函数的时候,就不能用void了。具体需要什么,看你需要返回什么值,整数就用int,其他跟变量名类似。需要返回啥值就用啥定义就好了。
Swap1就是这个函数的函数名了,函数名的命名要求也跟变量名的要求类似。
int x,int y。这个是函数的参数。用来接收主函数传递过来的值,也就是形参。(名称可以跟实参相同)
花括号{}里面的就是函数体了。
ok,大致介绍了一下函数,让我们来到今天的重点:形参和实参。
写一个函数实现交换两个整型变量的内容。
//错误的示范
void Swap1(int x, int y)
{
int tmp = x;
x = y;
y = tmp;
}
int main()
{
int a = 0;
int b = 0;
scanf("%d%d", &a, &b);
printf("交换前:a=%d b=%d\n", a, b);
Swap1(a, b);
printf("交换后:a=%d b=%d\n", a, b);
return 0;
}
上面这样看是不是感觉没什么问题,也能编译成功。值传进去,交换完之后回到主函数。再打印出来。让我们看看它的运行结果。
我们可以看到这根本没有完成交换。那么这是为什么呢。
这里就是跟形参和实参有关联了。形参上面介绍过了,这里面的形参就是(int x,int y)。而实参就是Swap1括号里面的(a,b)了。那么形参和实参是什么关系,又有什么区别呢。
这幅图我们观察到,在我们实参传递值给函数,也就是给形参。这个过程相当于我又开辟了一块空间将实参的值复制了一份过去,除了值一样以外,就跟实参没什么关系了。并不是说我把实参传进去,就是直接把这整个值给腾挪到一个新的地方,最后再挪回来。通俗点讲就是,我跟你买了一个一样的手机,无论我在我手机上的做任何操作都不会反馈到你的手机上。
可以看到这里面形参和实参的地址完全就不一样,也可以证明它们不是同一个东西。
那么我们是否就无法使用函数去完成这个要求呢,非也。
可以看到这个函数和上面那个函数的区别,(int* pa,int* pd),(&a,&b)。这是实参和形参的变化,为什么这样的变化让它们成功交换了呢。
上面我们说到,形参是实参的复制,是一块新开辟的空间,它们之间最大的隔阂就是地址不一样,无法影响到实参。那么我们直接用形参去找到实参的地址,再进行操作是不是就可以改变实参的值了。说到这里,你是不是也想到了,c语言中对地址进行操作的,是不是指针以及解引用*,取地址符&。
所以说这里面具体的流程是这样的,Swap2(&a,&b),我们把a和b的地址传了进去,函数里面用指针(int* pa,int* pd)进行接收,接收到地址之后,进行解引用的操作*pa,*pb,这时候这两个值就是a和b的值了。再进行交换的操作,进行交换的就是实实在在的a和b了。
解引用就是根据地址找到地址里面的内容,形象点讲就是根据你提供的你家的地址,我去找你一样。所以说找到你之后,在你家进行的操作都会影响到你家一样。
图片可以更加直观一点看出。整个流程的过程。
总的来说,我们把实参传进去的是值还是址进行区分。只是让形参复制了一份值,不对实参进行改变的叫做传值调用。
把传进去是地址,可以对实参进行改变的叫做传址调用。
这是我对实参和形参的理解,如果有什么错误或者讲的不清楚的地方,欢迎指出。