标签:c语言
普通函数参数
下面程序试图改变main函数中a和b的值
#include<stdio.h>
void fun(int x,int y)
{
int c;
c=a;
a=b;
b=c;
}
int main()
{
int a=1,b=2;
fun(a,b);
printf("a=%d b=%d\n",a,b);
return 0;
}
函数参数传递分析:
实参:
&a 28ff18 |
a 1 |
&b 28ff14 |
b 2 |
在将实参传递给形参的时候,编译器将实参拷贝一份给形参。
拷贝份的特点为:
1) 值与实参相同
2) 地址与实参不同
拷贝份(形参)
&x 28ff04 |
x 1 |
&y 28ff00 |
y 2 |
则在函数内部更改拷贝份的时候,只是对函数内部的局部变量改变,而局部变量是存储在栈空间里的,当fun函数执行结束后,这两个变量会自动销毁,不会改变实参的数值。
一级指针作为函数参数
下面程序试图改变main函数中a和b的值
#include<stdio.h>
void fun(int *a,int *b)
{
int c;
c=*a;
*a=*b;
*b=c;
}
int main()
{
int *a,*b;
int x=1,y=2;
a=&x;
b=&y;
fun(a,b);
printf("a=%d b=%d\n",x,y);
return 0;
}
函数参数传递分析
main中变量的存储情况如下
变量地址 |
变量值 |
&x: 28ff18 |
x:1 |
&y: 28ff14 |
y:2 |
指针变量地址 |
指针变量内容 |
指针表示地址里的内容 |
&a:28ff28 |
a: &x |
*a: x |
&b:28ff24 |
b: &y |
*b: y |
可见在main函数中,指针a和b的地址为28ff28,28ff24。而指针a和b存储的内容为x和y的地址28ff18、28ff14。*a,*b则是代表x和y变量的值。
当调用fun函数时。main中的指针a和b则要拷贝一份给fun函数的形参。则形参的存储情况如下:
形参指针变量地址 |
形参指针变量内容 |
形参指针变量存储的地址表示的内容 |
&a:28f740 |
a: 28ff18 |
*a: x |
&b:28f736 |
b: 28ff14 |
*b: y |
可见:把实参指针变量传递给形参后。
实参指针和形参指针的地址是不相同的。但是存储的内容确实相同的。均是指向main函数中的x,y变量,当再fun函数中执行c=*a;*a=*b;*b=c;交换语句时。*a和*b为main函数中的x、y变量。这时候将会交换这两个变量的值。
二级指针作为函数参数
如果想在函数中改变父函数中的指针变量呢。
#include<stdio.h>
void fun(int **c)
{
int *b;
b=(int *)malloc(sizeof(int));
*b=2;
*c=b;
}
int main()
{
int *a,x=1;
a=&x;
printf("%x\n",a);
fun(&a);
printf("%x\n",a);
return 0;
}
函数的实参存储如下:
指针变量地址 |
指针变量值 |
指针变量值表示的内容 |
&a:28f73b |
a:28f740 |
*a: x 1 |
变量地址 |
变量值 |
&x:28f740 |
x: 1 |
函数的形参存储如下
&c |
c:(&a) |
*c:(a) |
**c:(*a=x) |
28ff1c |
28f73b |
28f740 |
1 |
局部变量b的存储如下
&b |
b(堆空间地址) |
*b |
28ffeec |
391798 |
2 |
函数运行完后,指针变量a的指向则为在fun函数中申请的堆空间的地址,存储如下
指针变量地址 |
指针变量值 |
指针变量值表示的内容 |
&a:28f73b |
a:381798 |
*a: 2 |
将一级指针的地址赋值给形参的话,将会将一级指针的地址,变量值都赋值给二级指针的形参。此时在函数内部,二级指针(c)则能后获得外部一级指针变量(a)的地址、变量值、和普通变量(i)的值。只要知道实参变量的地址,则在函数内部就可以改变实参变量的值。所以这里的二级指针就可以改变一级指针所指向的内容(一级指针变量的值),一级指针所指向地址的内容。函数运行完后,一级指针a则指向申请的堆空间的地址,指向地址的内容则为堆空间存储的内容,不再指向变量i。
总结:在函数内部,形参能够获得实参变量的地址,则就能够改变实参的内容。
标签:c语言
原文地址:http://blog.csdn.net/u014104588/article/details/45457319