这里写一个Swap交换函数,目的为交换a,b的值
第一版:
void Swap(int a, int b)
{
int tmp = a;
a = b;
b = tmp;
}
int main()
{
int a = 10;
int b = 20;
printf("交换前:a=%d,b=%d\n",a,b);
Swap(a,b);
printf("交换后:a=%d,b=%d\n", a, b);
return 0;
}
程序执行为:
:
可以看出数值并没有交换,这里是为什么呢?
原因在于,形参里的值只是传入实参的一个副本,实参和形参隶属与两个函数空间,所以函数内修改的也是实参的副本,当然对实参没有如何影响了。
得出结论:值传递修改的是实参的副本
如何改正?
这里就要用到指针了
第二版:传指针
void Swap(int* p1, int* p2)
{
int* tmp = p1;
p1 = p2;
p2 = tmp;
}
int main()
{
int a = 10;
int b = 20;
printf("交换前:a=%d,b=%d\n",a,b);
Swap(&a,&b);
printf("交换后:a=%d,b=%d\n", a, b);
return 0;
}
程序执行结果为:
这里为什么还没有交换呢?原因在于没有解引用,仅仅交换了交换指针而不是指针所指向的变量的值,做的完全是无用功。
解引用非常重要
第三版:传指针,解引用
void Swap(int* p1, int* p2)
{
int* tmp;//tmp的值是随机值或者-858993460(0xcccccccc)
*tmp = *p1;//错误
*p1 = *p2;
*p2 = *tmp;
}
int main()
{
int a = 10;
int b = 20;
printf("交换前:a=%d,b=%d\n",a,b);
Swap(&a,&b);
printf("交换后:a=%d,b=%d\n", a, b);
return 0;
}
这段函数会程序编译出错 , 老的编译器 (vs2012,DEVC++,VC6.0 等 ) 程序能运行但在 *tmp=*p1; 这句崩溃
因为int* tmp时没有指定地址。
这里引入概念:野指针
此时tmp就是一个野指针,也称悬空指针,悬挂指针,会随机指向其他地址。问题就在于没有访问权限。就像你在小区随意去开其他人的门,当然不允许你进入。
再次改正
第四版:传指针,解引用
void Swap(int* p1, int* p2)
{
int tmp;
tmp = *p1;
*p1 = *p2;
*p2 = tmp;
}
int main()
{
int a = 10;
int b = 20;
printf("交换前:a=%d,b=%d\n",a,b);
Swap(&a,&b);
printf("交换后:a=%d,b=%d\n", a, b);
return 0;
}
此时终于运行成功!
结果:
画图分析为:
结论:调用函数通过形参想修改实参的值,必须1,传指针;2.解引用