在C语言中,函数传参有两种方式,传值和传地址,这俩者到底有何不同呢?
传值
传值是形参对实参的一份拷贝,如果在函数中对形参进行修改是不会影响外部的实参。
示例代码
void Swap1(int l, int r)//形参
{
int temp = l;
l = r;
r = temp;
}
int main()
{
int a = 10;
int b = 20;
Swap1(a, b);
printf("%d,%d\n", a, b);
system("pause");
return 0;
}
在监视中可以看出,形参l和r只是对实参a和b的一份拷贝,形参数值的改变并不影响实参的本质,俩组参数的地址始终不同。
传地址
传地址是指对函数中形参指向的内容进行修改,其外部的实参也会随之改变。实际上就是对地址的改变。
示例代码
void Swap2(int* pl, int* pr)
{
int temp = *pl;
*pl = *pr;
*pr = temp;
}
在传地址中,pl是指向a的指针,pr是指向b的指针,当修改pl和pr时,实际上是修改pl和pr所指向的内存空间的值,所以当pl和pr进行交换时,a、b也会随之改变。此时的pl和pr只是外部实参的一份拷贝。
注意
当改变函数中形参指针的指向,则不会影响实参
void Swap3(int* pl, int* pr)
{
int* temp = pl;
pl = pr;
pr = temp;
}
int main()
{
int a = 10;
int b = 20;
int*pa = &a;
int*pb = &b;
/*Swap1(a, b);
Swap2(&a, &b);*/
Swap3(pa, pb);//通过Swap3来改变pa和pb的指向
printf("%d,%d\n", a, b);
system("pause");
return 0;
}
指针pa指向a的地址,指针pb指向b的地址,当调用Swap3函数交换pa和pb的指向时,形参pl和pr也随之交换了指针的指向,但实参pa和pb的地址并没有交换,这就说明,当改变形参指针的指向时,并不影响外部实参。
那么在这种情况下,要改变实参该怎么做呢?我们可以给俩个二级指针ppl和ppr,ppl是指向内存地址pa的指针,ppr是指向内存地址pb的指针,当形参ppl和ppr发生改变时,其指向的内存空间也会发生改变。
示例代码
void Swap4(int** ppl, int** ppr)
{
int* temp = *ppl;
*ppl = *ppr;
*ppr = temp;
}
Swap4(&pa, &pb);
调式过程如下
在执行Swap4函数前,ppl是指向pa内存空间的指针,ppr是指向pb内存空间的指针,当改变形参ppl和ppr指向的内存空间时,实参pa和pb也发生改变。
总结
如果在函数体中,想要通过形参改变实参,则必须传递实参的地址。如果实参是变量,则要传递变量的地址,使用一级指针,如Swap2;如果实参是指针,则要传递指向指针的指针,使用二级指针,如Swap4;当在函数体中只是修改了指针的指向,则形参的改变不会影响外部实参,只是对实参的一份拷贝。