本文主要是介绍函数形参传递概念及问题分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
普通函数参数
下面程序试图改变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。
指针数组作为函数参数
下面程序是利用指针数组改变数组的值
#include<stdio.h>
void fun(int *c[],int n)
{
int i;
for(i=0;i<n;i++){
*c[i]=0;
}
}
int main()
{
int b[3]={1,2,3};
int *a[3]={b,b+1,b+2};
fun(a,3);
return 0;
}
在main函数中变量的存储情况如下
地址 | 内容 |
&b[0]:28ff04 | b[0]:1 |
&b[1]:28ff08 | b[1]:2 |
&b[2]:28ff0c | b[2]:3 |
地址 | 内容 | 指向内容 | ||
a:28fef8 | a[0]:28ff04 | *a[0] | ||
a+1:28fefc | a[1]:28ff08 | |||
a+2:28ff00 | a[2]:28ff0c |
这篇关于函数形参传递概念及问题分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!