c语⾔函数怎么调⽤数组_C语⾔函数学习-数组作为函数参数
8.1 数组作为函数参数
数组可以作为函数的参数使⽤,进⾏数据传送。数组⽤作函数参数有两种形式,⼀种是把数组元素(下标变量)作为实参使⽤;另⼀种是把数组名作为函数的形参和实参使⽤。
1. 数组元素作函数实参
数组元素就是下标变量,它与普通变量并⽆区别。 因此它作为函数实参使⽤与普通变量是完全相同的,
在发⽣函数调⽤时,把作为实参的数组元素的值传送给形参,实现单向的值传送。例5.4说明了这种情况。
【例8.7】判别⼀个整数数组中各元素的值,若⼤于0 则输出该值,若⼩于等于0则输出0值。编程如下:
void nzp(int v)
{
if(v>0)
printf("%d ",v);
else
printf("%d ",0);
}
main()
{
int a[5],i;
printf("input 5 numbers");
for(i=0;i<5;i++)
{scanf("%d",&a[i]);
nzp(a[i]);}
}
本程序中⾸先定义⼀个⽆返回值函数nzp,并说明其形参v为整型变量。在函数体中根据v值输出相应的结果。在main函数中⽤⼀个for语句输⼊数组各元素,每输⼊⼀个就以该元素作实参调⽤⼀次nzp函数,即把a[i]的值传送给形参v,供nzp函数使⽤。
2. 数组名作为函数参数
⽤数组名作函数参数与⽤数组元素作实参有⼏点不同:
scanf输入整型数组
1) ⽤数组元素作实参时,只要数组类型和函数的形参变量的类型⼀致,那么作为下标变量的数组元素的类型也和函数形参变量的类型是⼀致的。因此,并不要求函数的形参也是下标变量。换句话说,对数组元素的处理是按普通变量对待的。⽤数组名作函数参数时,则要求形参和相对应的实参都必须是类型相同的数组,都必须有明确的数组说明。当形参和实参⼆者不⼀致时,即会发⽣错误。
2) 在普通变量或下标变量作函数参数时,形参变量和实参变量是由编译系统分配的两个不同的内存单元。在函数调⽤时发⽣的值传送是把实参变量的值赋予形参变量。在⽤数组名作函数参数时,不是进⾏值的传送,即不是把实参数组的每⼀个元素的值都赋予形参数组的各个元素。因为实际上形参数组并不存在,编译系统不为形参数组分配内存。那么,数据的传送是如何实现的呢?在我们曾介绍过,数组名就是数组的⾸地址。因此在数组名作函数参数时所进⾏的传送只是地址的传送,也就是说把实参数组的⾸地址赋予形参数组名。形参数组名取得该⾸地址之后,也就等于有了实在的数组。实际上是形参数组和实参数组为同⼀数组,共同拥有⼀段内存空间。
上图说明了这种情形。图中设a为实参数组,类型为整型。a占有以2000为⾸地址的⼀块内存区。b为形参数组名。当发⽣函数调⽤时,进⾏地址传送,把实参数组a的⾸地址传送给形参数组名b,于是b也
取得该地址2000。于是a,b两数组共同占有以2000为⾸地址的⼀段连续内存单元。从图中还可以看出a和b下标相同的元素实际上也占相同的两个内存单元(整型数组每个元素占⼆字节)。例如a[0]和b[0]都占⽤2000和2001单元,当然a[0]等于b[0]。类推则有a[i]等于b[i]。
【例8.8】数组a中存放了⼀个学⽣5门课程的成绩,求平均成绩。
float aver(float a[5])
{
int i;
float av,s=a[0];
for(i=1;i<5;i++)
s=s+a[i];
av=s/5;
return av;
}
void main()
{
float sco[5],av;
int i;
printf("input 5 scores:");
for(i=0;i<5;i++)
scanf("%f",&sco[i]);
av=aver(sco);
printf("average score is %5.2f",av);
}
本程序⾸先定义了⼀个实型函数aver,有⼀个形参为实型数组a,长度为5。在函数aver中,把各元素值相加求出平均值,返回给主函数。主函数main 中⾸先完成数组sco的输⼊,然后以sco作为实参调⽤aver函数,函数返回值送av,最后输出av值。 从运⾏情况可以看出,程序实现了所要求的功能。
1) 前⾯已经讨论过,在变量作函数参数时,所进⾏的值传送是单向的。即只能从实参传向形参,不能从形参传回实参。形参的初值和实参相同,⽽形参的值发⽣改变后,实参并不变化,两者的终值是不同的。⽽当⽤数组名作函数参数时,情况则不同。由于实际上形参和实参为同⼀数组,因此当形参数组发⽣变化时,实参数组也随之变化。当然这种情况不能理解为发⽣了“双向”的值传递。但从实际情况来看,调⽤函数之后实参数组的值将由于形参数组值的变化⽽变化。为了说明这种情况,把例5.4改为例5.6的形式。
【例8.9】题⽬同8.7例。改⽤数组名作函数参数。
void nzp(int a[5])
{
int i;
printf("values of array a are:");
for(i=0;i<5;i++)
{
if(a[i]<0) a[i]=0;
printf("%d ",a[i]);
}
}
main()
{
int b[5],i;
printf("input 5 numbers:");
for(i=0;i<5;i++)
scanf("%d",&b[i]);
printf("initial values of array b are:");
for(i=0;i<5;i++)
printf("%d ",b[i]);
nzp(b);
printf("last values of array b are:");
for(i=0;i<5;i++)
printf("%d ",b[i]);
}
本程序中函数nzp的形参为整数组a,长度为5。主函数中实参数组b也为整型,长度也为5。在主函数中⾸先输⼊数组b的值,然后输出数组b的初始值。然后以数组名b为实参调⽤nzp函数。在nzp中,按要求把负值单元清0,并输出形参数组a的值。 返回主函数之后,再次输出数组b的值。从运⾏结果可
以看出,数组b的初值和终值是不同的,数组b的终值和数组a是相同的。这说明实参形参为同⼀数组,它们的值同时得以改变。
⽤数组名作为函数参数时还应注意以下⼏点:
a. 形参数组和实参数组的类型必须⼀致,否则将引起错误。
b. 形参数组和实参数组的长度可以不相同,因为在调⽤时,只传送⾸地址⽽不检查形参数组的长度。当形参数组的长度与实参数组不⼀致时,虽不⾄于出现语法错误(编译能通过),但程序执⾏结果将与实际不符,这是应予以注意的。
【例8.10】如把例8.9修改如下:
void nzp(int a[8])
{
int i;
printf("values of array aare:");
for(i=0;i<8;i++)
{
if(a[i]<0)a[i]=0;
printf("%d ",a[i]);
}
}
main()
{
int b[5],i;
printf("input 5 numbers:");
for(i=0;i<5;i++)
scanf("%d",&b[i]);
printf("initial values of array b are:");
for(i=0;i<5;i++)
printf("%d ",b[i]);
nzp(b);
printf("last values of array b are:");
for(i=0;i<5;i++)
printf("%d ",b[i]);
}
本程序与例8.9程序⽐,nzp函数的形参数组长度改为8,函数体中,for语句的循环条件也改为i<8。因此,形参数组a和实参数组b的长度不⼀致。编译能够通过,但从结果看,数组a的元素a[5],a[6],a[7]显然是⽆意义的。
c. 在函数形参表中,允许不给出形参数组的长度,或⽤⼀个变量来表⽰数组元素的个数。
例如,可以写为:
void nzp(int a[])
或写为
void nzp(int a[],int n)
其中形参数组a没有给出长度,⽽由n值动态地表⽰数组的长度。n的值由主调函数的实参进⾏传送。
由此,例8.10⼜可改为例8.11的形式。
【例8.11】
void nzp(int a[],int n)
{
int i;
printf("values of array a are:");
for(i=0;i
{
if(a[i]<0) a[i]=0;
printf("%d ",a[i]);
}
}
main()
{
int b[5],i;
printf("input 5 numbers:");
for(i=0;i<5;i++)
scanf("%d",&b[i]);
printf("initial values of array b are:");
for(i=0;i<5;i++)
printf("%d ",b[i]);
nzp(b,5);
printf("last values of array b are:");
for(i=0;i<5;i++)
printf("%d ",b[i]);
}
本程序nzp函数形参数组a没有给出长度,由n 动态确定该长度。在main函数中,函数调⽤语句为nzp(b,5),其中实参5将赋予形参n作为形参数组的长度。
d. 多维数组也可以作为函数的参数。在函数定义时对形参数组可以指定每⼀维的长度,也可省去第⼀维的长度。因此,以下写法都是合法的。
int MA(int a[3][10])
int MA(int a[][10])。

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。