计算机等级 > 二级考试 > C语言程序设计 > C语言辅导 > 文章内容

全国计算机等级考试二级C语言入门教程(74)

2016-2-16编辑:ljnbset

数组名作为函数参数

用数组名作函数参数与用数组元素作实参有几点不同:

1. 用数组元素作实参时,只要数组类型和函数的形参变量的类型一致,那么作为下标变量的数组元素的类型也和函数
形参变量的类型是一致的。因此, 并不要求函数的形参也是下标变量。 换句话说,对数组元素的处理是按普通变量
对待的。用数组名作函数参数时, 则要求形参和相对应的实参都必须是类型相同的数组,都必须有明确的数组说明。
当形参和实参二者不一致时,即会发生错误。
2. 在普通变量或下标变量作函数参数时,形参变量和实参变量是由编译系统分配的两个不同的内存单元。在函数调用
时发生的值传送是把实参变量的值赋予形参变量。在用数组名作函数参数时,不是进行值的传送,即不是把实参数组
的每一个元素的值都赋予形参数组的各个元素。因为实际上形参数组并不存在,编译系统不为形参数组分配内存。那
么,数据的传送是如何实现的呢? 在第四章中我们曾介绍过,数组名就是数组的首地址。因此在数组名作函数参数时
所进行的传送只是地址的传送, 也就是说把实参数组的首地址赋予形参数组名。形参数组名取得该首地址之后,也就
等于有了实在的数组。实际上是形参数组和实参数组为同一数组,共同拥有一段内存空间。图 5.1说明了这种情形。
图中设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]。
[例5.5]数组 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("\ninput 5 scores:\n");
for(i=0;i<5;i++)
scanf("%f",&sco[i]);
av=aver(sco);
printf("average score is %5.2f",av);
}



float aver(float a[5])

{ ……

}

void main()

{

……

for(i=0;i<5;i++)

scanf("%f",&sco[i]);

av=aver(sco);

……

}
本程序首先定义了一个实型函数aver,有一个形参为实型数组a,长度为 5。在函数 aver中,把各元素值相加求

出平均值,返回给主函数。主函数main 中首先完成数组 sco的输入,然后以 sco作为实参调用 aver函数,函数返回

值送av,最后输出 av值。 从运行情况可以看出,程序实现了所要求的功能

3. 前面已经讨论过,在变量作函数参数时,所进行的值传送是单向的。即只能从实参传向形参,不能从形参传回实参。
形参的初值和实参相同, 而形参的值发生改变后,实参并不变化, 两者的终值是不同的。例5.3证实了这个结论。 而
当用数组名作函数参数时,情况则不同。 由于实际上形参和实参为同一数组, 因此当形参数组发生变化时,实参数
组也随之变化。当然这种情况不能理解为发生了“双向”的值传递。但从实际情况来看,调用函数之后实参数组的值
将由于形参数组值的变化而变化。为了说明这种情况,把例5.4改为例 5.6的形式。[例5.6]题目同5.4例。改用数
组名作函数参数。
void nzp(int a[5])

{

int i;

printf("\nvalues of array a are:\n");

for(i=0;i<5;i++)

{

if(a[i]<0) a[i]=0;

printf("%d ",a[i]);

}

}

main()

{

int b[5],i;

printf("\ninput 5 numbers:\n");

for(i=0;i<5;i++)

scanf("%d",&b[i]);

printf("initial values of array b are:\n");

for(i=0;i<5;i++)

printf("%d ",b[i]);

nzp(b);

printf("\nlast values of array b are:\n");

for(i=0;i<5;i++)

printf("%d ",b[i]);

}

void nzp(int a[5])

{ ……

}

main()

{

int b[5],i;

……

nzp(b);

……

}
本程序中函数nzp的形参为整数组a,长度为 5。主函数中实参数组b也为整型,长度也为5。在主函数中首先

输入数组b的值,然后输出数组b的初始值。然后以数组名b为实参调用nzp函数。在nzp中,按要求把负值单元清

0,并输出形参数组 a的值。 返回主函数之后,再次输出数组b的值。从运行结果可以看出,数组b 的初值和终值是


不同的,数组b 的终值和数组 a是相同的。这说明实参形参为同一数组,它们的值同时得以改变。

用数组名作为函
数参数时还应注意以下几点:

a. 形参数组和实参数组的类型必须一致,否则将引起错误。
b. 形参数组和实参数组的长度可以不相同,因为在调用时,只传送首地址而不检查形参数组的长度。当形参数组的长
度与实参数组不一致时,虽不至于出现语法错误(编译能通过),但程序执行结果将与实际不符,这是应予以注意的。
如把例5.6修改如下:
void nzp(int a[8])

{

int i;

printf("\nvalues of array aare:\n");

for(i=0;i<8;i++)

{

if(a[i]<0)a[i]=0;

printf("%d",a[i]);

}

}

main()

{

int b[5],i;

printf("\ninput 5 numbers:\n");

for(i=0;i<5;i++)

scanf("%d",&b[i]);

printf("initial values of array b are:\n");

for(i=0;i<5;i++)

printf("%d",b[i]);

nzp(b);

printf("\nlast values of array b are:\n");

for(i=0;i<5;i++)

printf("%d",b[i]);

}

本程序与例 5.6程序比,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的值由主调函数的实参进行传送。
由此,例 5.6又可改为例 5.7的形式。

[例5.7]void nzp(int a[],int n)
{
int i;
printf("\nvalues of array a are:\n");
for(i=0;i{
if(a[i]<0) a[i]=0;
printf("%d ",a[i]);
}
}
main()
{
int b[5],i;
printf("\ninput 5 numbers:\n");
for(i=0;i<5;i++)
scanf("%d",&b[i]);
printf("initial values of array b are:\n");
for(i=0;i<5;i++)
printf("%d ",b[i]);



nzp(b,5);

printf("\nlast values of array b are:\n");

for(i=0;i<5;i++)

printf("%d ",b[i]);

}

void nzp(int a[],int n)

{ ……

}

main()

{

……

nzp(b,5);

……

}
本程序nzp函数形参数组 a没有给出长度,由n 动态确定该长度。在main函数中,函数调用语句为nzp(b,5),

其中实参 5将赋予形参 n作为形参数组的长度。

d. 多维数组也可以作为函数的参数。 在函数定义时对形参数组可以指定每一维的长度,也可省去第一维的长度。因
此,以下写法都是合法的。
int MA(int a[3][10])

int MA(int a[][10])

2017上半年全国计算机二级考试报名专题

热点推荐

登录注册
触屏版电脑版网站地图